198c485eaSDaniel Golle // SPDX-License-Identifier: GPL-2.0+
298c485eaSDaniel Golle #include <linux/bitfield.h>
3c66937b0SDaniel Golle #include <linux/bitmap.h>
4c66937b0SDaniel Golle #include <linux/mfd/syscon.h>
598c485eaSDaniel Golle #include <linux/module.h>
698c485eaSDaniel Golle #include <linux/nvmem-consumer.h>
798c485eaSDaniel Golle #include <linux/pinctrl/consumer.h>
898c485eaSDaniel Golle #include <linux/phy.h>
9c66937b0SDaniel Golle #include <linux/regmap.h>
1015d7b3dfSFrank Wunderlich #include <linux/of.h>
1198c485eaSDaniel Golle
12dc5a6164SHeiner Kallweit #include "../phylib.h"
137f9c320cSSkyLake.Huang #include "mtk.h"
147f9c320cSSkyLake.Huang
156a325aedSChristian Marangi #define MTK_PHY_MAX_LEDS 2
166a325aedSChristian Marangi
1798c485eaSDaniel Golle #define MTK_GPHY_ID_MT7981 0x03a29461
1898c485eaSDaniel Golle #define MTK_GPHY_ID_MT7988 0x03a29481
196a325aedSChristian Marangi #define MTK_GPHY_ID_AN7581 0x03a294c1
20*8bc3c234SChristian Marangi #define MTK_GPHY_ID_AN7583 0xc0ff0420
2198c485eaSDaniel Golle
2298c485eaSDaniel Golle #define MTK_EXT_PAGE_ACCESS 0x1f
2398c485eaSDaniel Golle #define MTK_PHY_PAGE_STANDARD 0x0000
2498c485eaSDaniel Golle #define MTK_PHY_PAGE_EXTENDED_3 0x0003
2598c485eaSDaniel Golle
2698c485eaSDaniel Golle #define MTK_PHY_LPI_REG_14 0x14
2798c485eaSDaniel Golle #define MTK_PHY_LPI_WAKE_TIMER_1000_MASK GENMASK(8, 0)
2898c485eaSDaniel Golle
2998c485eaSDaniel Golle #define MTK_PHY_LPI_REG_1c 0x1c
3098c485eaSDaniel Golle #define MTK_PHY_SMI_DET_ON_THRESH_MASK GENMASK(13, 8)
3198c485eaSDaniel Golle
3298c485eaSDaniel Golle #define MTK_PHY_PAGE_EXTENDED_2A30 0x2a30
3398c485eaSDaniel Golle
34afa08fdeSSky Huang /* Registers on Token Ring debug nodes */
35afa08fdeSSky Huang /* ch_addr = 0x0, node_addr = 0x7, data_addr = 0x15 */
36afa08fdeSSky Huang /* NormMseLoThresh */
37afa08fdeSSky Huang #define NORMAL_MSE_LO_THRESH_MASK GENMASK(15, 8)
38afa08fdeSSky Huang
39afa08fdeSSky Huang /* ch_addr = 0x0, node_addr = 0xf, data_addr = 0x3c */
40afa08fdeSSky Huang /* RemAckCntLimitCtrl */
41afa08fdeSSky Huang #define REMOTE_ACK_COUNT_LIMIT_CTRL_MASK GENMASK(2, 1)
42afa08fdeSSky Huang
43afa08fdeSSky Huang /* ch_addr = 0x1, node_addr = 0xd, data_addr = 0x20 */
44afa08fdeSSky Huang /* VcoSlicerThreshBitsHigh */
45afa08fdeSSky Huang #define VCO_SLICER_THRESH_HIGH_MASK GENMASK(23, 0)
46afa08fdeSSky Huang
47afa08fdeSSky Huang /* ch_addr = 0x1, node_addr = 0xf, data_addr = 0x0 */
48afa08fdeSSky Huang /* DfeTailEnableVgaThresh1000 */
49afa08fdeSSky Huang #define DFE_TAIL_EANBLE_VGA_TRHESH_1000 GENMASK(5, 1)
50afa08fdeSSky Huang
51afa08fdeSSky Huang /* ch_addr = 0x1, node_addr = 0xf, data_addr = 0x1 */
52afa08fdeSSky Huang /* MrvlTrFix100Kp */
53afa08fdeSSky Huang #define MRVL_TR_FIX_100KP_MASK GENMASK(22, 20)
54afa08fdeSSky Huang /* MrvlTrFix100Kf */
55afa08fdeSSky Huang #define MRVL_TR_FIX_100KF_MASK GENMASK(19, 17)
56afa08fdeSSky Huang /* MrvlTrFix1000Kp */
57afa08fdeSSky Huang #define MRVL_TR_FIX_1000KP_MASK GENMASK(16, 14)
58afa08fdeSSky Huang /* MrvlTrFix1000Kf */
59afa08fdeSSky Huang #define MRVL_TR_FIX_1000KF_MASK GENMASK(13, 11)
60afa08fdeSSky Huang
61afa08fdeSSky Huang /* ch_addr = 0x1, node_addr = 0xf, data_addr = 0x12 */
62afa08fdeSSky Huang /* VgaDecRate */
63afa08fdeSSky Huang #define VGA_DECIMATION_RATE_MASK GENMASK(8, 5)
64afa08fdeSSky Huang
65afa08fdeSSky Huang /* ch_addr = 0x1, node_addr = 0xf, data_addr = 0x17 */
66afa08fdeSSky Huang /* SlvDSPreadyTime */
67afa08fdeSSky Huang #define SLAVE_DSP_READY_TIME_MASK GENMASK(22, 15)
68afa08fdeSSky Huang /* MasDSPreadyTime */
69afa08fdeSSky Huang #define MASTER_DSP_READY_TIME_MASK GENMASK(14, 7)
70afa08fdeSSky Huang
7140d33d6dSSky Huang /* ch_addr = 0x1, node_addr = 0xf, data_addr = 0x18 */
7240d33d6dSSky Huang /* EnabRandUpdTrig */
7340d33d6dSSky Huang #define ENABLE_RANDOM_UPDOWN_COUNTER_TRIGGER BIT(8)
7440d33d6dSSky Huang
75afa08fdeSSky Huang /* ch_addr = 0x1, node_addr = 0xf, data_addr = 0x20 */
76afa08fdeSSky Huang /* ResetSyncOffset */
77afa08fdeSSky Huang #define RESET_SYNC_OFFSET_MASK GENMASK(11, 8)
78afa08fdeSSky Huang
79afa08fdeSSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x0 */
80afa08fdeSSky Huang /* FfeUpdGainForceVal */
81afa08fdeSSky Huang #define FFE_UPDATE_GAIN_FORCE_VAL_MASK GENMASK(9, 7)
82afa08fdeSSky Huang /* FfeUpdGainForce */
83afa08fdeSSky Huang #define FFE_UPDATE_GAIN_FORCE BIT(6)
84afa08fdeSSky Huang
854786eff2SSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x3 */
864786eff2SSky Huang /* TrFreeze */
874786eff2SSky Huang #define TR_FREEZE_MASK GENMASK(11, 0)
884786eff2SSky Huang
89afa08fdeSSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x6 */
90afa08fdeSSky Huang /* SS: Steady-state, KP: Proportional Gain */
91afa08fdeSSky Huang /* SSTrKp100 */
92afa08fdeSSky Huang #define SS_TR_KP100_MASK GENMASK(21, 19)
93afa08fdeSSky Huang /* SSTrKf100 */
94afa08fdeSSky Huang #define SS_TR_KF100_MASK GENMASK(18, 16)
95afa08fdeSSky Huang /* SSTrKp1000Mas */
96afa08fdeSSky Huang #define SS_TR_KP1000_MASTER_MASK GENMASK(15, 13)
97afa08fdeSSky Huang /* SSTrKf1000Mas */
98afa08fdeSSky Huang #define SS_TR_KF1000_MASTER_MASK GENMASK(12, 10)
99afa08fdeSSky Huang /* SSTrKp1000Slv */
100afa08fdeSSky Huang #define SS_TR_KP1000_SLAVE_MASK GENMASK(9, 7)
101afa08fdeSSky Huang /* SSTrKf1000Slv */
102afa08fdeSSky Huang #define SS_TR_KF1000_SLAVE_MASK GENMASK(6, 4)
103afa08fdeSSky Huang
1044786eff2SSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x8 */
1054786eff2SSky Huang /* clear this bit if wanna select from AFE */
1064786eff2SSky Huang /* Regsigdet_sel_1000 */
1074786eff2SSky Huang #define EEE1000_SELECT_SIGNAL_DETECTION_FROM_DFE BIT(4)
1084786eff2SSky Huang
109afa08fdeSSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0xd */
110afa08fdeSSky Huang /* RegEEE_st2TrKf1000 */
111afa08fdeSSky Huang #define EEE1000_STAGE2_TR_KF_MASK GENMASK(13, 11)
112afa08fdeSSky Huang
113afa08fdeSSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0xf */
114afa08fdeSSky Huang /* RegEEE_slv_waketr_timer_tar */
115afa08fdeSSky Huang #define SLAVE_WAKETR_TIMER_MASK GENMASK(20, 11)
116afa08fdeSSky Huang /* RegEEE_slv_remtx_timer_tar */
117afa08fdeSSky Huang #define SLAVE_REMTX_TIMER_MASK GENMASK(10, 1)
118afa08fdeSSky Huang
119afa08fdeSSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x10 */
120afa08fdeSSky Huang /* RegEEE_slv_wake_int_timer_tar */
121afa08fdeSSky Huang #define SLAVE_WAKEINT_TIMER_MASK GENMASK(10, 1)
122afa08fdeSSky Huang
123afa08fdeSSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x14 */
124afa08fdeSSky Huang /* RegEEE_trfreeze_timer2 */
125afa08fdeSSky Huang #define TR_FREEZE_TIMER2_MASK GENMASK(9, 0)
126afa08fdeSSky Huang
127afa08fdeSSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x1c */
128afa08fdeSSky Huang /* RegEEE100Stg1_tar */
129afa08fdeSSky Huang #define EEE100_LPSYNC_STAGE1_UPDATE_TIMER_MASK GENMASK(8, 0)
130afa08fdeSSky Huang
1314786eff2SSky Huang /* ch_addr = 0x2, node_addr = 0xd, data_addr = 0x25 */
1324786eff2SSky Huang /* REGEEE_wake_slv_tr_wait_dfesigdet_en */
1334786eff2SSky Huang #define WAKE_SLAVE_TR_WAIT_DFE_DETECTION_EN BIT(11)
1344786eff2SSky Huang
13598c485eaSDaniel Golle #define ANALOG_INTERNAL_OPERATION_MAX_US 20
13698c485eaSDaniel Golle #define TXRESERVE_MIN 0
13798c485eaSDaniel Golle #define TXRESERVE_MAX 7
13898c485eaSDaniel Golle
13998c485eaSDaniel Golle #define MTK_PHY_ANARG_RG 0x10
14098c485eaSDaniel Golle #define MTK_PHY_TCLKOFFSET_MASK GENMASK(12, 8)
14198c485eaSDaniel Golle
14298c485eaSDaniel Golle /* Registers on MDIO_MMD_VEND1 */
14398c485eaSDaniel Golle #define MTK_PHY_TXVLD_DA_RG 0x12
14498c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_A_GBE_MASK GENMASK(15, 10)
14598c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_A_TBT_MASK GENMASK(5, 0)
14698c485eaSDaniel Golle
14798c485eaSDaniel Golle #define MTK_PHY_TX_I2MPB_TEST_MODE_A2 0x16
14898c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_A_HBT_MASK GENMASK(15, 10)
14998c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_A_TST_MASK GENMASK(5, 0)
15098c485eaSDaniel Golle
15198c485eaSDaniel Golle #define MTK_PHY_TX_I2MPB_TEST_MODE_B1 0x17
15298c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_B_GBE_MASK GENMASK(13, 8)
15398c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_B_TBT_MASK GENMASK(5, 0)
15498c485eaSDaniel Golle
15598c485eaSDaniel Golle #define MTK_PHY_TX_I2MPB_TEST_MODE_B2 0x18
15698c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_B_HBT_MASK GENMASK(13, 8)
15798c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_B_TST_MASK GENMASK(5, 0)
15898c485eaSDaniel Golle
15998c485eaSDaniel Golle #define MTK_PHY_TX_I2MPB_TEST_MODE_C1 0x19
16098c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_C_GBE_MASK GENMASK(13, 8)
16198c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_C_TBT_MASK GENMASK(5, 0)
16298c485eaSDaniel Golle
16398c485eaSDaniel Golle #define MTK_PHY_TX_I2MPB_TEST_MODE_C2 0x20
16498c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_C_HBT_MASK GENMASK(13, 8)
16598c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_C_TST_MASK GENMASK(5, 0)
16698c485eaSDaniel Golle
16798c485eaSDaniel Golle #define MTK_PHY_TX_I2MPB_TEST_MODE_D1 0x21
16898c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_D_GBE_MASK GENMASK(13, 8)
16998c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_D_TBT_MASK GENMASK(5, 0)
17098c485eaSDaniel Golle
17198c485eaSDaniel Golle #define MTK_PHY_TX_I2MPB_TEST_MODE_D2 0x22
17298c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_D_HBT_MASK GENMASK(13, 8)
17398c485eaSDaniel Golle #define MTK_PHY_DA_TX_I2MPB_D_TST_MASK GENMASK(5, 0)
17498c485eaSDaniel Golle
17598c485eaSDaniel Golle #define MTK_PHY_RXADC_CTRL_RG7 0xc6
17698c485eaSDaniel Golle #define MTK_PHY_DA_AD_BUF_BIAS_LP_MASK GENMASK(9, 8)
17798c485eaSDaniel Golle
17898c485eaSDaniel Golle #define MTK_PHY_RXADC_CTRL_RG9 0xc8
17998c485eaSDaniel Golle #define MTK_PHY_DA_RX_PSBN_TBT_MASK GENMASK(14, 12)
18098c485eaSDaniel Golle #define MTK_PHY_DA_RX_PSBN_HBT_MASK GENMASK(10, 8)
18198c485eaSDaniel Golle #define MTK_PHY_DA_RX_PSBN_GBE_MASK GENMASK(6, 4)
18298c485eaSDaniel Golle #define MTK_PHY_DA_RX_PSBN_LP_MASK GENMASK(2, 0)
18398c485eaSDaniel Golle
18498c485eaSDaniel Golle #define MTK_PHY_LDO_OUTPUT_V 0xd7
18598c485eaSDaniel Golle
18698c485eaSDaniel Golle #define MTK_PHY_RG_ANA_CAL_RG0 0xdb
18798c485eaSDaniel Golle #define MTK_PHY_RG_CAL_CKINV BIT(12)
18898c485eaSDaniel Golle #define MTK_PHY_RG_ANA_CALEN BIT(8)
18998c485eaSDaniel Golle #define MTK_PHY_RG_ZCALEN_A BIT(0)
19098c485eaSDaniel Golle
19198c485eaSDaniel Golle #define MTK_PHY_RG_ANA_CAL_RG1 0xdc
19298c485eaSDaniel Golle #define MTK_PHY_RG_ZCALEN_B BIT(12)
19398c485eaSDaniel Golle #define MTK_PHY_RG_ZCALEN_C BIT(8)
19498c485eaSDaniel Golle #define MTK_PHY_RG_ZCALEN_D BIT(4)
19598c485eaSDaniel Golle #define MTK_PHY_RG_TXVOS_CALEN BIT(0)
19698c485eaSDaniel Golle
19798c485eaSDaniel Golle #define MTK_PHY_RG_ANA_CAL_RG5 0xe0
19898c485eaSDaniel Golle #define MTK_PHY_RG_REXT_TRIM_MASK GENMASK(13, 8)
19998c485eaSDaniel Golle
20098c485eaSDaniel Golle #define MTK_PHY_RG_TX_FILTER 0xfe
20198c485eaSDaniel Golle
20298c485eaSDaniel Golle #define MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG120 0x120
20398c485eaSDaniel Golle #define MTK_PHY_LPI_SIG_EN_LO_THRESH1000_MASK GENMASK(12, 8)
20498c485eaSDaniel Golle #define MTK_PHY_LPI_SIG_EN_HI_THRESH1000_MASK GENMASK(4, 0)
20598c485eaSDaniel Golle
20698c485eaSDaniel Golle #define MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG122 0x122
20798c485eaSDaniel Golle #define MTK_PHY_LPI_NORM_MSE_HI_THRESH1000_MASK GENMASK(7, 0)
20898c485eaSDaniel Golle
20998c485eaSDaniel Golle #define MTK_PHY_RG_TESTMUX_ADC_CTRL 0x144
21098c485eaSDaniel Golle #define MTK_PHY_RG_TXEN_DIG_MASK GENMASK(5, 5)
21198c485eaSDaniel Golle
21298c485eaSDaniel Golle #define MTK_PHY_RG_CR_TX_AMP_OFFSET_A_B 0x172
21398c485eaSDaniel Golle #define MTK_PHY_CR_TX_AMP_OFFSET_A_MASK GENMASK(13, 8)
21498c485eaSDaniel Golle #define MTK_PHY_CR_TX_AMP_OFFSET_B_MASK GENMASK(6, 0)
21598c485eaSDaniel Golle
21698c485eaSDaniel Golle #define MTK_PHY_RG_CR_TX_AMP_OFFSET_C_D 0x173
21798c485eaSDaniel Golle #define MTK_PHY_CR_TX_AMP_OFFSET_C_MASK GENMASK(13, 8)
21898c485eaSDaniel Golle #define MTK_PHY_CR_TX_AMP_OFFSET_D_MASK GENMASK(6, 0)
21998c485eaSDaniel Golle
22098c485eaSDaniel Golle #define MTK_PHY_RG_AD_CAL_COMP 0x17a
22193a610c0SSkyLake.Huang #define MTK_PHY_AD_CAL_COMP_OUT_MASK GENMASK(8, 8)
22298c485eaSDaniel Golle
22398c485eaSDaniel Golle #define MTK_PHY_RG_AD_CAL_CLK 0x17b
22498c485eaSDaniel Golle #define MTK_PHY_DA_CAL_CLK BIT(0)
22598c485eaSDaniel Golle
22698c485eaSDaniel Golle #define MTK_PHY_RG_AD_CALIN 0x17c
22798c485eaSDaniel Golle #define MTK_PHY_DA_CALIN_FLAG BIT(0)
22898c485eaSDaniel Golle
22998c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN0_A 0x17d
23098c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN0_A_MASK GENMASK(9, 0)
23198c485eaSDaniel Golle
23298c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN0_B 0x17e
23398c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN0_B_MASK GENMASK(9, 0)
23498c485eaSDaniel Golle
23598c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN0_C 0x17f
23698c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN0_C_MASK GENMASK(9, 0)
23798c485eaSDaniel Golle
23898c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN0_D 0x180
23998c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN0_D_MASK GENMASK(9, 0)
24098c485eaSDaniel Golle
24198c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN1_A 0x181
24298c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN1_A_MASK GENMASK(9, 0)
24398c485eaSDaniel Golle
24498c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN1_B 0x182
24598c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN1_B_MASK GENMASK(9, 0)
24698c485eaSDaniel Golle
24798c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN1_C 0x183
24898c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN1_C_MASK GENMASK(9, 0)
24998c485eaSDaniel Golle
25098c485eaSDaniel Golle #define MTK_PHY_RG_DASN_DAC_IN1_D 0x184
25198c485eaSDaniel Golle #define MTK_PHY_DASN_DAC_IN1_D_MASK GENMASK(9, 0)
25298c485eaSDaniel Golle
25398c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG19b 0x19b
25498c485eaSDaniel Golle #define MTK_PHY_BYPASS_DSP_LPI_READY BIT(8)
25598c485eaSDaniel Golle
25698c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K1_L 0x22a
25798c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K1_U 0x22b
25898c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K2_L 0x22c
25998c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K2_U 0x22d
26098c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K3_L 0x22e
26198c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K3_U 0x22f
26298c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K4_L 0x230
26398c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K4_U 0x231
26498c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K5_L 0x232
26598c485eaSDaniel Golle #define MTK_PHY_RG_LP_IIR2_K5_U 0x233
26698c485eaSDaniel Golle
26798c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG234 0x234
26898c485eaSDaniel Golle #define MTK_PHY_TR_OPEN_LOOP_EN_MASK GENMASK(0, 0)
26998c485eaSDaniel Golle #define MTK_PHY_LPF_X_AVERAGE_MASK GENMASK(7, 4)
27098c485eaSDaniel Golle #define MTK_PHY_TR_LP_IIR_EEE_EN BIT(12)
27198c485eaSDaniel Golle
27298c485eaSDaniel Golle #define MTK_PHY_RG_LPF_CNT_VAL 0x235
27398c485eaSDaniel Golle
27498c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG238 0x238
27598c485eaSDaniel Golle #define MTK_PHY_LPI_SLV_SEND_TX_TIMER_MASK GENMASK(8, 0)
27698c485eaSDaniel Golle #define MTK_PHY_LPI_SLV_SEND_TX_EN BIT(12)
27798c485eaSDaniel Golle
27898c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG239 0x239
27998c485eaSDaniel Golle #define MTK_PHY_LPI_SEND_LOC_TIMER_MASK GENMASK(8, 0)
28098c485eaSDaniel Golle #define MTK_PHY_LPI_TXPCS_LOC_RCV BIT(12)
28198c485eaSDaniel Golle
28298c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG27C 0x27c
28398c485eaSDaniel Golle #define MTK_PHY_VGASTATE_FFE_THR_ST1_MASK GENMASK(12, 8)
28498c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG27D 0x27d
28598c485eaSDaniel Golle #define MTK_PHY_VGASTATE_FFE_THR_ST2_MASK GENMASK(4, 0)
28698c485eaSDaniel Golle
28798c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG2C7 0x2c7
28898c485eaSDaniel Golle #define MTK_PHY_MAX_GAIN_MASK GENMASK(4, 0)
28998c485eaSDaniel Golle #define MTK_PHY_MIN_GAIN_MASK GENMASK(12, 8)
29098c485eaSDaniel Golle
29198c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG2D1 0x2d1
29298c485eaSDaniel Golle #define MTK_PHY_VCO_SLICER_THRESH_BITS_HIGH_EEE_MASK GENMASK(7, 0)
29398c485eaSDaniel Golle #define MTK_PHY_LPI_SKIP_SD_SLV_TR BIT(8)
29498c485eaSDaniel Golle #define MTK_PHY_LPI_TR_READY BIT(9)
29598c485eaSDaniel Golle #define MTK_PHY_LPI_VCO_EEE_STG0_EN BIT(10)
29698c485eaSDaniel Golle
29798c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG323 0x323
29898c485eaSDaniel Golle #define MTK_PHY_EEE_WAKE_MAS_INT_DC BIT(0)
29998c485eaSDaniel Golle #define MTK_PHY_EEE_WAKE_SLV_INT_DC BIT(4)
30098c485eaSDaniel Golle
30198c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG324 0x324
30298c485eaSDaniel Golle #define MTK_PHY_SMI_DETCNT_MAX_MASK GENMASK(5, 0)
30398c485eaSDaniel Golle #define MTK_PHY_SMI_DET_MAX_EN BIT(8)
30498c485eaSDaniel Golle
30598c485eaSDaniel Golle #define MTK_PHY_RG_DEV1E_REG326 0x326
30698c485eaSDaniel Golle #define MTK_PHY_LPI_MODE_SD_ON BIT(0)
30798c485eaSDaniel Golle #define MTK_PHY_RESET_RANDUPD_CNT BIT(1)
30898c485eaSDaniel Golle #define MTK_PHY_TREC_UPDATE_ENAB_CLR BIT(2)
30998c485eaSDaniel Golle #define MTK_PHY_LPI_QUIT_WAIT_DFE_SIG_DET_OFF BIT(4)
31098c485eaSDaniel Golle #define MTK_PHY_TR_READY_SKIP_AFE_WAKEUP BIT(5)
31198c485eaSDaniel Golle
31298c485eaSDaniel Golle #define MTK_PHY_LDO_PUMP_EN_PAIRAB 0x502
31398c485eaSDaniel Golle #define MTK_PHY_LDO_PUMP_EN_PAIRCD 0x503
31498c485eaSDaniel Golle
31598c485eaSDaniel Golle #define MTK_PHY_DA_TX_R50_PAIR_A 0x53d
31698c485eaSDaniel Golle #define MTK_PHY_DA_TX_R50_PAIR_B 0x53e
31798c485eaSDaniel Golle #define MTK_PHY_DA_TX_R50_PAIR_C 0x53f
31898c485eaSDaniel Golle #define MTK_PHY_DA_TX_R50_PAIR_D 0x540
31998c485eaSDaniel Golle
320c66937b0SDaniel Golle /* Registers on MDIO_MMD_VEND2 */
321c66937b0SDaniel Golle #define MTK_PHY_LED1_DEFAULT_POLARITIES BIT(1)
322c66937b0SDaniel Golle
32398c485eaSDaniel Golle #define MTK_PHY_RG_BG_RASEL 0x115
32498c485eaSDaniel Golle #define MTK_PHY_RG_BG_RASEL_MASK GENMASK(2, 0)
32598c485eaSDaniel Golle
326c66937b0SDaniel Golle /* 'boottrap' register reflecting the configuration of the 4 PHY LEDs */
327c66937b0SDaniel Golle #define RG_GPIO_MISC_TPBANK0 0x6f0
328c66937b0SDaniel Golle #define RG_GPIO_MISC_TPBANK0_BOOTMODE GENMASK(11, 8)
329c66937b0SDaniel Golle
33098c485eaSDaniel Golle /* These macro privides efuse parsing for internal phy. */
33198c485eaSDaniel Golle #define EFS_DA_TX_I2MPB_A(x) (((x) >> 0) & GENMASK(5, 0))
33298c485eaSDaniel Golle #define EFS_DA_TX_I2MPB_B(x) (((x) >> 6) & GENMASK(5, 0))
33398c485eaSDaniel Golle #define EFS_DA_TX_I2MPB_C(x) (((x) >> 12) & GENMASK(5, 0))
33498c485eaSDaniel Golle #define EFS_DA_TX_I2MPB_D(x) (((x) >> 18) & GENMASK(5, 0))
33598c485eaSDaniel Golle #define EFS_DA_TX_AMP_OFFSET_A(x) (((x) >> 24) & GENMASK(5, 0))
33698c485eaSDaniel Golle
33798c485eaSDaniel Golle #define EFS_DA_TX_AMP_OFFSET_B(x) (((x) >> 0) & GENMASK(5, 0))
33898c485eaSDaniel Golle #define EFS_DA_TX_AMP_OFFSET_C(x) (((x) >> 6) & GENMASK(5, 0))
33998c485eaSDaniel Golle #define EFS_DA_TX_AMP_OFFSET_D(x) (((x) >> 12) & GENMASK(5, 0))
34098c485eaSDaniel Golle #define EFS_DA_TX_R50_A(x) (((x) >> 18) & GENMASK(5, 0))
34198c485eaSDaniel Golle #define EFS_DA_TX_R50_B(x) (((x) >> 24) & GENMASK(5, 0))
34298c485eaSDaniel Golle
34398c485eaSDaniel Golle #define EFS_DA_TX_R50_C(x) (((x) >> 0) & GENMASK(5, 0))
34498c485eaSDaniel Golle #define EFS_DA_TX_R50_D(x) (((x) >> 6) & GENMASK(5, 0))
34598c485eaSDaniel Golle
34698c485eaSDaniel Golle #define EFS_RG_BG_RASEL(x) (((x) >> 4) & GENMASK(2, 0))
34798c485eaSDaniel Golle #define EFS_RG_REXT_TRIM(x) (((x) >> 7) & GENMASK(5, 0))
34898c485eaSDaniel Golle
34998c485eaSDaniel Golle enum {
35098c485eaSDaniel Golle NO_PAIR,
35198c485eaSDaniel Golle PAIR_A,
35298c485eaSDaniel Golle PAIR_B,
35398c485eaSDaniel Golle PAIR_C,
35498c485eaSDaniel Golle PAIR_D,
35598c485eaSDaniel Golle };
35698c485eaSDaniel Golle
35798c485eaSDaniel Golle enum calibration_mode {
35898c485eaSDaniel Golle EFUSE_K,
35998c485eaSDaniel Golle SW_K
36098c485eaSDaniel Golle };
36198c485eaSDaniel Golle
36298c485eaSDaniel Golle enum CAL_ITEM {
36398c485eaSDaniel Golle REXT,
36498c485eaSDaniel Golle TX_OFFSET,
36598c485eaSDaniel Golle TX_AMP,
36698c485eaSDaniel Golle TX_R50,
36798c485eaSDaniel Golle TX_VCM
36898c485eaSDaniel Golle };
36998c485eaSDaniel Golle
37098c485eaSDaniel Golle enum CAL_MODE {
37198c485eaSDaniel Golle EFUSE_M,
37298c485eaSDaniel Golle SW_M
37398c485eaSDaniel Golle };
37498c485eaSDaniel Golle
375c66937b0SDaniel Golle struct mtk_socphy_shared {
376c66937b0SDaniel Golle u32 boottrap;
377c66937b0SDaniel Golle struct mtk_socphy_priv priv[4];
378c66937b0SDaniel Golle };
379c66937b0SDaniel Golle
38098c485eaSDaniel Golle /* One calibration cycle consists of:
38198c485eaSDaniel Golle * 1.Set DA_CALIN_FLAG high to start calibration. Keep it high
38298c485eaSDaniel Golle * until AD_CAL_COMP is ready to output calibration result.
38398c485eaSDaniel Golle * 2.Wait until DA_CAL_CLK is available.
38498c485eaSDaniel Golle * 3.Fetch AD_CAL_COMP_OUT.
38598c485eaSDaniel Golle */
cal_cycle(struct phy_device * phydev,int devad,u32 regnum,u16 mask,u16 cal_val)38698c485eaSDaniel Golle static int cal_cycle(struct phy_device *phydev, int devad,
38798c485eaSDaniel Golle u32 regnum, u16 mask, u16 cal_val)
38898c485eaSDaniel Golle {
38998c485eaSDaniel Golle int reg_val;
39098c485eaSDaniel Golle int ret;
39198c485eaSDaniel Golle
39298c485eaSDaniel Golle phy_modify_mmd(phydev, devad, regnum,
39398c485eaSDaniel Golle mask, cal_val);
39498c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_AD_CALIN,
39598c485eaSDaniel Golle MTK_PHY_DA_CALIN_FLAG);
39698c485eaSDaniel Golle
39798c485eaSDaniel Golle ret = phy_read_mmd_poll_timeout(phydev, MDIO_MMD_VEND1,
39898c485eaSDaniel Golle MTK_PHY_RG_AD_CAL_CLK, reg_val,
39998c485eaSDaniel Golle reg_val & MTK_PHY_DA_CAL_CLK, 500,
400b0f90a86SSkyLake.Huang ANALOG_INTERNAL_OPERATION_MAX_US,
401b0f90a86SSkyLake.Huang false);
40298c485eaSDaniel Golle if (ret) {
40398c485eaSDaniel Golle phydev_err(phydev, "Calibration cycle timeout\n");
40498c485eaSDaniel Golle return ret;
40598c485eaSDaniel Golle }
40698c485eaSDaniel Golle
40798c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_AD_CALIN,
40898c485eaSDaniel Golle MTK_PHY_DA_CALIN_FLAG);
40993a610c0SSkyLake.Huang ret = phy_read_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_AD_CAL_COMP);
41093a610c0SSkyLake.Huang if (ret < 0)
41193a610c0SSkyLake.Huang return ret;
41293a610c0SSkyLake.Huang ret = FIELD_GET(MTK_PHY_AD_CAL_COMP_OUT_MASK, ret);
41398c485eaSDaniel Golle phydev_dbg(phydev, "cal_val: 0x%x, ret: %d\n", cal_val, ret);
41498c485eaSDaniel Golle
41598c485eaSDaniel Golle return ret;
41698c485eaSDaniel Golle }
41798c485eaSDaniel Golle
rext_fill_result(struct phy_device * phydev,u16 * buf)41898c485eaSDaniel Golle static int rext_fill_result(struct phy_device *phydev, u16 *buf)
41998c485eaSDaniel Golle {
42098c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG5,
42198c485eaSDaniel Golle MTK_PHY_RG_REXT_TRIM_MASK, buf[0] << 8);
42298c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND2, MTK_PHY_RG_BG_RASEL,
42398c485eaSDaniel Golle MTK_PHY_RG_BG_RASEL_MASK, buf[1]);
42498c485eaSDaniel Golle
42598c485eaSDaniel Golle return 0;
42698c485eaSDaniel Golle }
42798c485eaSDaniel Golle
rext_cal_efuse(struct phy_device * phydev,u32 * buf)42898c485eaSDaniel Golle static int rext_cal_efuse(struct phy_device *phydev, u32 *buf)
42998c485eaSDaniel Golle {
43098c485eaSDaniel Golle u16 rext_cal_val[2];
43198c485eaSDaniel Golle
43298c485eaSDaniel Golle rext_cal_val[0] = EFS_RG_REXT_TRIM(buf[3]);
43398c485eaSDaniel Golle rext_cal_val[1] = EFS_RG_BG_RASEL(buf[3]);
43498c485eaSDaniel Golle rext_fill_result(phydev, rext_cal_val);
43598c485eaSDaniel Golle
43698c485eaSDaniel Golle return 0;
43798c485eaSDaniel Golle }
43898c485eaSDaniel Golle
tx_offset_fill_result(struct phy_device * phydev,u16 * buf)43998c485eaSDaniel Golle static int tx_offset_fill_result(struct phy_device *phydev, u16 *buf)
44098c485eaSDaniel Golle {
44198c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_CR_TX_AMP_OFFSET_A_B,
44298c485eaSDaniel Golle MTK_PHY_CR_TX_AMP_OFFSET_A_MASK, buf[0] << 8);
44398c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_CR_TX_AMP_OFFSET_A_B,
44498c485eaSDaniel Golle MTK_PHY_CR_TX_AMP_OFFSET_B_MASK, buf[1]);
44598c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_CR_TX_AMP_OFFSET_C_D,
44698c485eaSDaniel Golle MTK_PHY_CR_TX_AMP_OFFSET_C_MASK, buf[2] << 8);
44798c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_CR_TX_AMP_OFFSET_C_D,
44898c485eaSDaniel Golle MTK_PHY_CR_TX_AMP_OFFSET_D_MASK, buf[3]);
44998c485eaSDaniel Golle
45098c485eaSDaniel Golle return 0;
45198c485eaSDaniel Golle }
45298c485eaSDaniel Golle
tx_offset_cal_efuse(struct phy_device * phydev,u32 * buf)45398c485eaSDaniel Golle static int tx_offset_cal_efuse(struct phy_device *phydev, u32 *buf)
45498c485eaSDaniel Golle {
45598c485eaSDaniel Golle u16 tx_offset_cal_val[4];
45698c485eaSDaniel Golle
45798c485eaSDaniel Golle tx_offset_cal_val[0] = EFS_DA_TX_AMP_OFFSET_A(buf[0]);
45898c485eaSDaniel Golle tx_offset_cal_val[1] = EFS_DA_TX_AMP_OFFSET_B(buf[1]);
45998c485eaSDaniel Golle tx_offset_cal_val[2] = EFS_DA_TX_AMP_OFFSET_C(buf[1]);
46098c485eaSDaniel Golle tx_offset_cal_val[3] = EFS_DA_TX_AMP_OFFSET_D(buf[1]);
46198c485eaSDaniel Golle
46298c485eaSDaniel Golle tx_offset_fill_result(phydev, tx_offset_cal_val);
46398c485eaSDaniel Golle
46498c485eaSDaniel Golle return 0;
46598c485eaSDaniel Golle }
46698c485eaSDaniel Golle
tx_amp_fill_result(struct phy_device * phydev,u16 * buf)46798c485eaSDaniel Golle static int tx_amp_fill_result(struct phy_device *phydev, u16 *buf)
46898c485eaSDaniel Golle {
46998c485eaSDaniel Golle const int vals_9481[16] = { 10, 6, 6, 10,
47098c485eaSDaniel Golle 10, 6, 6, 10,
47198c485eaSDaniel Golle 10, 6, 6, 10,
47298c485eaSDaniel Golle 10, 6, 6, 10 };
473b544223bSSkyLake.Huang const int vals_9461[16] = { 7, 1, 4, 7,
474b544223bSSkyLake.Huang 7, 1, 4, 7,
475b544223bSSkyLake.Huang 7, 1, 4, 7,
476b544223bSSkyLake.Huang 7, 1, 4, 7 };
477b544223bSSkyLake.Huang int bias[16] = {};
478b544223bSSkyLake.Huang int i;
479b544223bSSkyLake.Huang
48098c485eaSDaniel Golle switch (phydev->drv->phy_id) {
48198c485eaSDaniel Golle case MTK_GPHY_ID_MT7981:
48298c485eaSDaniel Golle /* We add some calibration to efuse values
48398c485eaSDaniel Golle * due to board level influence.
48498c485eaSDaniel Golle * GBE: +7, TBT: +1, HBT: +4, TST: +7
48598c485eaSDaniel Golle */
48698c485eaSDaniel Golle memcpy(bias, (const void *)vals_9461, sizeof(bias));
48798c485eaSDaniel Golle break;
48898c485eaSDaniel Golle case MTK_GPHY_ID_MT7988:
48998c485eaSDaniel Golle memcpy(bias, (const void *)vals_9481, sizeof(bias));
49098c485eaSDaniel Golle break;
49198c485eaSDaniel Golle }
49298c485eaSDaniel Golle
49398c485eaSDaniel Golle /* Prevent overflow */
49498c485eaSDaniel Golle for (i = 0; i < 12; i++) {
49598c485eaSDaniel Golle if (buf[i >> 2] + bias[i] > 63) {
49698c485eaSDaniel Golle buf[i >> 2] = 63;
49798c485eaSDaniel Golle bias[i] = 0;
49898c485eaSDaniel Golle }
49998c485eaSDaniel Golle }
50098c485eaSDaniel Golle
50198c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TXVLD_DA_RG,
502b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_A_GBE_MASK,
503b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_A_GBE_MASK,
504b0f90a86SSkyLake.Huang buf[0] + bias[0]));
50598c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TXVLD_DA_RG,
506b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_A_TBT_MASK,
507b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_A_TBT_MASK,
508b0f90a86SSkyLake.Huang buf[0] + bias[1]));
50998c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_A2,
510b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_A_HBT_MASK,
511b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_A_HBT_MASK,
512b0f90a86SSkyLake.Huang buf[0] + bias[2]));
51398c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_A2,
514b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_A_TST_MASK,
515b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_A_TST_MASK,
516b0f90a86SSkyLake.Huang buf[0] + bias[3]));
51798c485eaSDaniel Golle
51898c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_B1,
519b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_B_GBE_MASK,
520b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_B_GBE_MASK,
521b0f90a86SSkyLake.Huang buf[1] + bias[4]));
52298c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_B1,
523b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_B_TBT_MASK,
524b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_B_TBT_MASK,
525b0f90a86SSkyLake.Huang buf[1] + bias[5]));
52698c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_B2,
527b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_B_HBT_MASK,
528b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_B_HBT_MASK,
529b0f90a86SSkyLake.Huang buf[1] + bias[6]));
53098c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_B2,
531b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_B_TST_MASK,
532b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_B_TST_MASK,
533b0f90a86SSkyLake.Huang buf[1] + bias[7]));
53498c485eaSDaniel Golle
53598c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_C1,
536b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_C_GBE_MASK,
537b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_C_GBE_MASK,
538b0f90a86SSkyLake.Huang buf[2] + bias[8]));
53998c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_C1,
540b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_C_TBT_MASK,
541b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_C_TBT_MASK,
542b0f90a86SSkyLake.Huang buf[2] + bias[9]));
54398c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_C2,
544b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_C_HBT_MASK,
545b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_C_HBT_MASK,
546b0f90a86SSkyLake.Huang buf[2] + bias[10]));
54798c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_C2,
548b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_C_TST_MASK,
549b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_C_TST_MASK,
550b0f90a86SSkyLake.Huang buf[2] + bias[11]));
55198c485eaSDaniel Golle
55298c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_D1,
553b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_D_GBE_MASK,
554b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_D_GBE_MASK,
555b0f90a86SSkyLake.Huang buf[3] + bias[12]));
55698c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_D1,
557b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_D_TBT_MASK,
558b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_D_TBT_MASK,
559b0f90a86SSkyLake.Huang buf[3] + bias[13]));
56098c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_D2,
561b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_D_HBT_MASK,
562b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_D_HBT_MASK,
563b0f90a86SSkyLake.Huang buf[3] + bias[14]));
56498c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_TX_I2MPB_TEST_MODE_D2,
565b0f90a86SSkyLake.Huang MTK_PHY_DA_TX_I2MPB_D_TST_MASK,
566b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_DA_TX_I2MPB_D_TST_MASK,
567b0f90a86SSkyLake.Huang buf[3] + bias[15]));
56898c485eaSDaniel Golle
56998c485eaSDaniel Golle return 0;
57098c485eaSDaniel Golle }
57198c485eaSDaniel Golle
tx_amp_cal_efuse(struct phy_device * phydev,u32 * buf)57298c485eaSDaniel Golle static int tx_amp_cal_efuse(struct phy_device *phydev, u32 *buf)
57398c485eaSDaniel Golle {
57498c485eaSDaniel Golle u16 tx_amp_cal_val[4];
57598c485eaSDaniel Golle
57698c485eaSDaniel Golle tx_amp_cal_val[0] = EFS_DA_TX_I2MPB_A(buf[0]);
57798c485eaSDaniel Golle tx_amp_cal_val[1] = EFS_DA_TX_I2MPB_B(buf[0]);
57898c485eaSDaniel Golle tx_amp_cal_val[2] = EFS_DA_TX_I2MPB_C(buf[0]);
57998c485eaSDaniel Golle tx_amp_cal_val[3] = EFS_DA_TX_I2MPB_D(buf[0]);
58098c485eaSDaniel Golle tx_amp_fill_result(phydev, tx_amp_cal_val);
58198c485eaSDaniel Golle
58298c485eaSDaniel Golle return 0;
58398c485eaSDaniel Golle }
58498c485eaSDaniel Golle
tx_r50_fill_result(struct phy_device * phydev,u16 tx_r50_cal_val,u8 txg_calen_x)58598c485eaSDaniel Golle static int tx_r50_fill_result(struct phy_device *phydev, u16 tx_r50_cal_val,
58698c485eaSDaniel Golle u8 txg_calen_x)
58798c485eaSDaniel Golle {
58898c485eaSDaniel Golle int bias = 0;
58998c485eaSDaniel Golle u16 reg, val;
59098c485eaSDaniel Golle
59198c485eaSDaniel Golle if (phydev->drv->phy_id == MTK_GPHY_ID_MT7988)
592ff63cc2eSDaniel Golle bias = -1;
59398c485eaSDaniel Golle
59498c485eaSDaniel Golle val = clamp_val(bias + tx_r50_cal_val, 0, 63);
59598c485eaSDaniel Golle
59698c485eaSDaniel Golle switch (txg_calen_x) {
59798c485eaSDaniel Golle case PAIR_A:
59898c485eaSDaniel Golle reg = MTK_PHY_DA_TX_R50_PAIR_A;
59998c485eaSDaniel Golle break;
60098c485eaSDaniel Golle case PAIR_B:
60198c485eaSDaniel Golle reg = MTK_PHY_DA_TX_R50_PAIR_B;
60298c485eaSDaniel Golle break;
60398c485eaSDaniel Golle case PAIR_C:
60498c485eaSDaniel Golle reg = MTK_PHY_DA_TX_R50_PAIR_C;
60598c485eaSDaniel Golle break;
60698c485eaSDaniel Golle case PAIR_D:
60798c485eaSDaniel Golle reg = MTK_PHY_DA_TX_R50_PAIR_D;
60898c485eaSDaniel Golle break;
60998c485eaSDaniel Golle default:
61098c485eaSDaniel Golle return -EINVAL;
61198c485eaSDaniel Golle }
61298c485eaSDaniel Golle
61398c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, reg, val | val << 8);
61498c485eaSDaniel Golle
61598c485eaSDaniel Golle return 0;
61698c485eaSDaniel Golle }
61798c485eaSDaniel Golle
tx_r50_cal_efuse(struct phy_device * phydev,u32 * buf,u8 txg_calen_x)61898c485eaSDaniel Golle static int tx_r50_cal_efuse(struct phy_device *phydev, u32 *buf,
61998c485eaSDaniel Golle u8 txg_calen_x)
62098c485eaSDaniel Golle {
62198c485eaSDaniel Golle u16 tx_r50_cal_val;
62298c485eaSDaniel Golle
62398c485eaSDaniel Golle switch (txg_calen_x) {
62498c485eaSDaniel Golle case PAIR_A:
62598c485eaSDaniel Golle tx_r50_cal_val = EFS_DA_TX_R50_A(buf[1]);
62698c485eaSDaniel Golle break;
62798c485eaSDaniel Golle case PAIR_B:
62898c485eaSDaniel Golle tx_r50_cal_val = EFS_DA_TX_R50_B(buf[1]);
62998c485eaSDaniel Golle break;
63098c485eaSDaniel Golle case PAIR_C:
63198c485eaSDaniel Golle tx_r50_cal_val = EFS_DA_TX_R50_C(buf[2]);
63298c485eaSDaniel Golle break;
63398c485eaSDaniel Golle case PAIR_D:
63498c485eaSDaniel Golle tx_r50_cal_val = EFS_DA_TX_R50_D(buf[2]);
63598c485eaSDaniel Golle break;
63698c485eaSDaniel Golle default:
63798c485eaSDaniel Golle return -EINVAL;
63898c485eaSDaniel Golle }
63998c485eaSDaniel Golle tx_r50_fill_result(phydev, tx_r50_cal_val, txg_calen_x);
64098c485eaSDaniel Golle
64198c485eaSDaniel Golle return 0;
64298c485eaSDaniel Golle }
64398c485eaSDaniel Golle
tx_vcm_cal_sw(struct phy_device * phydev,u8 rg_txreserve_x)64498c485eaSDaniel Golle static int tx_vcm_cal_sw(struct phy_device *phydev, u8 rg_txreserve_x)
64598c485eaSDaniel Golle {
64698c485eaSDaniel Golle u8 lower_idx, upper_idx, txreserve_val;
64798c485eaSDaniel Golle u8 lower_ret, upper_ret;
64898c485eaSDaniel Golle int ret;
64998c485eaSDaniel Golle
65098c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG0,
65198c485eaSDaniel Golle MTK_PHY_RG_ANA_CALEN);
65298c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG0,
65398c485eaSDaniel Golle MTK_PHY_RG_CAL_CKINV);
65498c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG1,
65598c485eaSDaniel Golle MTK_PHY_RG_TXVOS_CALEN);
65698c485eaSDaniel Golle
65798c485eaSDaniel Golle switch (rg_txreserve_x) {
65898c485eaSDaniel Golle case PAIR_A:
65998c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
66098c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN0_A,
66198c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN0_A_MASK);
66298c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
66398c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN1_A,
66498c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN1_A_MASK);
66598c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1,
66698c485eaSDaniel Golle MTK_PHY_RG_ANA_CAL_RG0,
66798c485eaSDaniel Golle MTK_PHY_RG_ZCALEN_A);
66898c485eaSDaniel Golle break;
66998c485eaSDaniel Golle case PAIR_B:
67098c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
67198c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN0_B,
67298c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN0_B_MASK);
67398c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
67498c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN1_B,
67598c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN1_B_MASK);
67698c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1,
67798c485eaSDaniel Golle MTK_PHY_RG_ANA_CAL_RG1,
67898c485eaSDaniel Golle MTK_PHY_RG_ZCALEN_B);
67998c485eaSDaniel Golle break;
68098c485eaSDaniel Golle case PAIR_C:
68198c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
68298c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN0_C,
68398c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN0_C_MASK);
68498c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
68598c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN1_C,
68698c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN1_C_MASK);
68798c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1,
68898c485eaSDaniel Golle MTK_PHY_RG_ANA_CAL_RG1,
68998c485eaSDaniel Golle MTK_PHY_RG_ZCALEN_C);
69098c485eaSDaniel Golle break;
69198c485eaSDaniel Golle case PAIR_D:
69298c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
69398c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN0_D,
69498c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN0_D_MASK);
69598c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
69698c485eaSDaniel Golle MTK_PHY_RG_DASN_DAC_IN1_D,
69798c485eaSDaniel Golle MTK_PHY_DASN_DAC_IN1_D_MASK);
69898c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1,
69998c485eaSDaniel Golle MTK_PHY_RG_ANA_CAL_RG1,
70098c485eaSDaniel Golle MTK_PHY_RG_ZCALEN_D);
70198c485eaSDaniel Golle break;
70298c485eaSDaniel Golle default:
70398c485eaSDaniel Golle ret = -EINVAL;
70498c485eaSDaniel Golle goto restore;
70598c485eaSDaniel Golle }
70698c485eaSDaniel Golle
70798c485eaSDaniel Golle lower_idx = TXRESERVE_MIN;
70898c485eaSDaniel Golle upper_idx = TXRESERVE_MAX;
70998c485eaSDaniel Golle
71098c485eaSDaniel Golle phydev_dbg(phydev, "Start TX-VCM SW cal.\n");
71198c485eaSDaniel Golle while ((upper_idx - lower_idx) > 1) {
71298c485eaSDaniel Golle txreserve_val = DIV_ROUND_CLOSEST(lower_idx + upper_idx, 2);
71398c485eaSDaniel Golle ret = cal_cycle(phydev, MDIO_MMD_VEND1, MTK_PHY_RXADC_CTRL_RG9,
71498c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_TBT_MASK |
71598c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_HBT_MASK |
71698c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_GBE_MASK |
71798c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_LP_MASK,
71898c485eaSDaniel Golle txreserve_val << 12 | txreserve_val << 8 |
71998c485eaSDaniel Golle txreserve_val << 4 | txreserve_val);
72098c485eaSDaniel Golle if (ret == 1) {
72198c485eaSDaniel Golle upper_idx = txreserve_val;
72298c485eaSDaniel Golle upper_ret = ret;
72398c485eaSDaniel Golle } else if (ret == 0) {
72498c485eaSDaniel Golle lower_idx = txreserve_val;
72598c485eaSDaniel Golle lower_ret = ret;
72698c485eaSDaniel Golle } else {
72798c485eaSDaniel Golle goto restore;
72898c485eaSDaniel Golle }
72998c485eaSDaniel Golle }
73098c485eaSDaniel Golle
73198c485eaSDaniel Golle if (lower_idx == TXRESERVE_MIN) {
73298c485eaSDaniel Golle lower_ret = cal_cycle(phydev, MDIO_MMD_VEND1,
73398c485eaSDaniel Golle MTK_PHY_RXADC_CTRL_RG9,
73498c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_TBT_MASK |
73598c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_HBT_MASK |
73698c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_GBE_MASK |
73798c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_LP_MASK,
73898c485eaSDaniel Golle lower_idx << 12 | lower_idx << 8 |
73998c485eaSDaniel Golle lower_idx << 4 | lower_idx);
74098c485eaSDaniel Golle ret = lower_ret;
74198c485eaSDaniel Golle } else if (upper_idx == TXRESERVE_MAX) {
74298c485eaSDaniel Golle upper_ret = cal_cycle(phydev, MDIO_MMD_VEND1,
74398c485eaSDaniel Golle MTK_PHY_RXADC_CTRL_RG9,
74498c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_TBT_MASK |
74598c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_HBT_MASK |
74698c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_GBE_MASK |
74798c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_LP_MASK,
74898c485eaSDaniel Golle upper_idx << 12 | upper_idx << 8 |
74998c485eaSDaniel Golle upper_idx << 4 | upper_idx);
75098c485eaSDaniel Golle ret = upper_ret;
75198c485eaSDaniel Golle }
75298c485eaSDaniel Golle if (ret < 0)
75398c485eaSDaniel Golle goto restore;
75498c485eaSDaniel Golle
75598c485eaSDaniel Golle /* We calibrate TX-VCM in different logic. Check upper index and then
756b0f90a86SSkyLake.Huang * lower index. If this calibration is valid, apply lower index's
757b0f90a86SSkyLake.Huang * result.
75898c485eaSDaniel Golle */
75998c485eaSDaniel Golle ret = upper_ret - lower_ret;
76098c485eaSDaniel Golle if (ret == 1) {
76198c485eaSDaniel Golle ret = 0;
76298c485eaSDaniel Golle /* Make sure we use upper_idx in our calibration system */
76398c485eaSDaniel Golle cal_cycle(phydev, MDIO_MMD_VEND1, MTK_PHY_RXADC_CTRL_RG9,
76498c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_TBT_MASK |
76598c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_HBT_MASK |
76698c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_GBE_MASK |
76798c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_LP_MASK,
76898c485eaSDaniel Golle upper_idx << 12 | upper_idx << 8 |
76998c485eaSDaniel Golle upper_idx << 4 | upper_idx);
77098c485eaSDaniel Golle phydev_dbg(phydev, "TX-VCM SW cal result: 0x%x\n", upper_idx);
77198c485eaSDaniel Golle } else if (lower_idx == TXRESERVE_MIN && upper_ret == 1 &&
77298c485eaSDaniel Golle lower_ret == 1) {
77398c485eaSDaniel Golle ret = 0;
77498c485eaSDaniel Golle cal_cycle(phydev, MDIO_MMD_VEND1, MTK_PHY_RXADC_CTRL_RG9,
77598c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_TBT_MASK |
77698c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_HBT_MASK |
77798c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_GBE_MASK |
77898c485eaSDaniel Golle MTK_PHY_DA_RX_PSBN_LP_MASK,
77998c485eaSDaniel Golle lower_idx << 12 | lower_idx << 8 |
78098c485eaSDaniel Golle lower_idx << 4 | lower_idx);
78198c485eaSDaniel Golle phydev_warn(phydev, "TX-VCM SW cal result at low margin 0x%x\n",
78298c485eaSDaniel Golle lower_idx);
78398c485eaSDaniel Golle } else if (upper_idx == TXRESERVE_MAX && upper_ret == 0 &&
78498c485eaSDaniel Golle lower_ret == 0) {
78598c485eaSDaniel Golle ret = 0;
786b0f90a86SSkyLake.Huang phydev_warn(phydev,
787b0f90a86SSkyLake.Huang "TX-VCM SW cal result at high margin 0x%x\n",
78898c485eaSDaniel Golle upper_idx);
78998c485eaSDaniel Golle } else {
79098c485eaSDaniel Golle ret = -EINVAL;
79198c485eaSDaniel Golle }
79298c485eaSDaniel Golle
79398c485eaSDaniel Golle restore:
79498c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG0,
79598c485eaSDaniel Golle MTK_PHY_RG_ANA_CALEN);
79698c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG1,
79798c485eaSDaniel Golle MTK_PHY_RG_TXVOS_CALEN);
79898c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG0,
79998c485eaSDaniel Golle MTK_PHY_RG_ZCALEN_A);
80098c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_ANA_CAL_RG1,
80198c485eaSDaniel Golle MTK_PHY_RG_ZCALEN_B | MTK_PHY_RG_ZCALEN_C |
80298c485eaSDaniel Golle MTK_PHY_RG_ZCALEN_D);
80398c485eaSDaniel Golle
80498c485eaSDaniel Golle return ret;
80598c485eaSDaniel Golle }
80698c485eaSDaniel Golle
mt798x_phy_common_finetune(struct phy_device * phydev)80798c485eaSDaniel Golle static void mt798x_phy_common_finetune(struct phy_device *phydev)
80898c485eaSDaniel Golle {
80998c485eaSDaniel Golle phy_select_page(phydev, MTK_PHY_PAGE_EXTENDED_52B5);
810afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xf, 0x17,
811afa08fdeSSky Huang SLAVE_DSP_READY_TIME_MASK | MASTER_DSP_READY_TIME_MASK,
812afa08fdeSSky Huang FIELD_PREP(SLAVE_DSP_READY_TIME_MASK, 0x18) |
813afa08fdeSSky Huang FIELD_PREP(MASTER_DSP_READY_TIME_MASK, 0x18));
814ff63cc2eSDaniel Golle
81540d33d6dSSky Huang __mtk_tr_set_bits(phydev, 0x1, 0xf, 0x18,
81640d33d6dSSky Huang ENABLE_RANDOM_UPDOWN_COUNTER_TRIGGER);
81798c485eaSDaniel Golle
818afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x0, 0x7, 0x15,
819afa08fdeSSky Huang NORMAL_MSE_LO_THRESH_MASK,
820afa08fdeSSky Huang FIELD_PREP(NORMAL_MSE_LO_THRESH_MASK, 0x55));
82198c485eaSDaniel Golle
822afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x2, 0xd, 0x0,
823afa08fdeSSky Huang FFE_UPDATE_GAIN_FORCE_VAL_MASK,
824afa08fdeSSky Huang FIELD_PREP(FFE_UPDATE_GAIN_FORCE_VAL_MASK, 0x4) |
825afa08fdeSSky Huang FFE_UPDATE_GAIN_FORCE);
826ff63cc2eSDaniel Golle
8274786eff2SSky Huang __mtk_tr_clr_bits(phydev, 0x2, 0xd, 0x3, TR_FREEZE_MASK);
82898c485eaSDaniel Golle
829afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x2, 0xd, 0x6,
830afa08fdeSSky Huang SS_TR_KP100_MASK | SS_TR_KF100_MASK |
831afa08fdeSSky Huang SS_TR_KP1000_MASTER_MASK | SS_TR_KF1000_MASTER_MASK |
832afa08fdeSSky Huang SS_TR_KP1000_SLAVE_MASK | SS_TR_KF1000_SLAVE_MASK,
833afa08fdeSSky Huang FIELD_PREP(SS_TR_KP100_MASK, 0x5) |
834afa08fdeSSky Huang FIELD_PREP(SS_TR_KF100_MASK, 0x6) |
835afa08fdeSSky Huang FIELD_PREP(SS_TR_KP1000_MASTER_MASK, 0x5) |
836afa08fdeSSky Huang FIELD_PREP(SS_TR_KF1000_MASTER_MASK, 0x6) |
837afa08fdeSSky Huang FIELD_PREP(SS_TR_KP1000_SLAVE_MASK, 0x5) |
838afa08fdeSSky Huang FIELD_PREP(SS_TR_KF1000_SLAVE_MASK, 0x6));
839afa08fdeSSky Huang
840ff63cc2eSDaniel Golle phy_restore_page(phydev, MTK_PHY_PAGE_STANDARD, 0);
841ff63cc2eSDaniel Golle }
842ff63cc2eSDaniel Golle
mt7981_phy_finetune(struct phy_device * phydev)843ff63cc2eSDaniel Golle static void mt7981_phy_finetune(struct phy_device *phydev)
844ff63cc2eSDaniel Golle {
845ff63cc2eSDaniel Golle u16 val[8] = { 0x01ce, 0x01c1,
846ff63cc2eSDaniel Golle 0x020f, 0x0202,
847ff63cc2eSDaniel Golle 0x03d0, 0x03c0,
848ff63cc2eSDaniel Golle 0x0013, 0x0005 };
849ff63cc2eSDaniel Golle int i, k;
850ff63cc2eSDaniel Golle
851ff63cc2eSDaniel Golle /* 100M eye finetune:
852ff63cc2eSDaniel Golle * Keep middle level of TX MLT3 shapper as default.
853ff63cc2eSDaniel Golle * Only change TX MLT3 overshoot level here.
854ff63cc2eSDaniel Golle */
855ff63cc2eSDaniel Golle for (k = 0, i = 1; i < 12; i++) {
856ff63cc2eSDaniel Golle if (i % 3 == 0)
857ff63cc2eSDaniel Golle continue;
858ff63cc2eSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, i, val[k++]);
859ff63cc2eSDaniel Golle }
860ff63cc2eSDaniel Golle
861ff63cc2eSDaniel Golle phy_select_page(phydev, MTK_PHY_PAGE_EXTENDED_52B5);
862afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xf, 0x20,
863afa08fdeSSky Huang RESET_SYNC_OFFSET_MASK,
864afa08fdeSSky Huang FIELD_PREP(RESET_SYNC_OFFSET_MASK, 0x6));
865ff63cc2eSDaniel Golle
866afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xf, 0x12,
867afa08fdeSSky Huang VGA_DECIMATION_RATE_MASK,
868afa08fdeSSky Huang FIELD_PREP(VGA_DECIMATION_RATE_MASK, 0x1));
86998c485eaSDaniel Golle
87098c485eaSDaniel Golle /* MrvlTrFix100Kp = 3, MrvlTrFix100Kf = 2,
87198c485eaSDaniel Golle * MrvlTrFix1000Kp = 3, MrvlTrFix1000Kf = 2
87298c485eaSDaniel Golle */
873afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xf, 0x1,
874afa08fdeSSky Huang MRVL_TR_FIX_100KP_MASK | MRVL_TR_FIX_100KF_MASK |
875afa08fdeSSky Huang MRVL_TR_FIX_1000KP_MASK | MRVL_TR_FIX_1000KF_MASK,
876afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_100KP_MASK, 0x3) |
877afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_100KF_MASK, 0x2) |
878afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_1000KP_MASK, 0x3) |
879afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_1000KF_MASK, 0x2));
88098c485eaSDaniel Golle
88198c485eaSDaniel Golle /* VcoSlicerThreshBitsHigh */
882afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xd, 0x20,
883afa08fdeSSky Huang VCO_SLICER_THRESH_HIGH_MASK,
884afa08fdeSSky Huang FIELD_PREP(VCO_SLICER_THRESH_HIGH_MASK, 0x555555));
88598c485eaSDaniel Golle phy_restore_page(phydev, MTK_PHY_PAGE_STANDARD, 0);
88698c485eaSDaniel Golle
88798c485eaSDaniel Golle /* TR_OPEN_LOOP_EN = 1, lpf_x_average = 9 */
88898c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG234,
889b0f90a86SSkyLake.Huang MTK_PHY_TR_OPEN_LOOP_EN_MASK |
890b0f90a86SSkyLake.Huang MTK_PHY_LPF_X_AVERAGE_MASK,
89198c485eaSDaniel Golle BIT(0) | FIELD_PREP(MTK_PHY_LPF_X_AVERAGE_MASK, 0x9));
89298c485eaSDaniel Golle
89398c485eaSDaniel Golle /* rg_tr_lpf_cnt_val = 512 */
89498c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LPF_CNT_VAL, 0x200);
89598c485eaSDaniel Golle
89698c485eaSDaniel Golle /* IIR2 related */
89798c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K1_L, 0x82);
89898c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K1_U, 0x0);
89998c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K2_L, 0x103);
90098c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K2_U, 0x0);
90198c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K3_L, 0x82);
90298c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K3_U, 0x0);
90398c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K4_L, 0xd177);
90498c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K4_U, 0x3);
90598c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K5_L, 0x2c82);
90698c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LP_IIR2_K5_U, 0xe);
90798c485eaSDaniel Golle
90898c485eaSDaniel Golle /* FFE peaking */
90998c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG27C,
91098c485eaSDaniel Golle MTK_PHY_VGASTATE_FFE_THR_ST1_MASK, 0x1b << 8);
91198c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG27D,
91298c485eaSDaniel Golle MTK_PHY_VGASTATE_FFE_THR_ST2_MASK, 0x1e);
91398c485eaSDaniel Golle
91498c485eaSDaniel Golle /* Disable LDO pump */
91598c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_LDO_PUMP_EN_PAIRAB, 0x0);
91698c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_LDO_PUMP_EN_PAIRCD, 0x0);
91798c485eaSDaniel Golle /* Adjust LDO output voltage */
91898c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_LDO_OUTPUT_V, 0x2222);
91998c485eaSDaniel Golle }
92098c485eaSDaniel Golle
mt7988_phy_finetune(struct phy_device * phydev)92198c485eaSDaniel Golle static void mt7988_phy_finetune(struct phy_device *phydev)
92298c485eaSDaniel Golle {
92398c485eaSDaniel Golle u16 val[12] = { 0x0187, 0x01cd, 0x01c8, 0x0182,
92498c485eaSDaniel Golle 0x020d, 0x0206, 0x0384, 0x03d0,
92598c485eaSDaniel Golle 0x03c6, 0x030a, 0x0011, 0x0005 };
92698c485eaSDaniel Golle int i;
92798c485eaSDaniel Golle
92898c485eaSDaniel Golle /* Set default MLT3 shaper first */
92998c485eaSDaniel Golle for (i = 0; i < 12; i++)
93098c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, i, val[i]);
93198c485eaSDaniel Golle
93298c485eaSDaniel Golle /* TCT finetune */
93398c485eaSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_TX_FILTER, 0x5);
93498c485eaSDaniel Golle
93598c485eaSDaniel Golle phy_select_page(phydev, MTK_PHY_PAGE_EXTENDED_52B5);
936afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xf, 0x20,
937afa08fdeSSky Huang RESET_SYNC_OFFSET_MASK,
938afa08fdeSSky Huang FIELD_PREP(RESET_SYNC_OFFSET_MASK, 0x5));
93998c485eaSDaniel Golle
94098c485eaSDaniel Golle /* VgaDecRate is 1 at default on mt7988 */
94198c485eaSDaniel Golle
942afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xf, 0x1,
943afa08fdeSSky Huang MRVL_TR_FIX_100KP_MASK | MRVL_TR_FIX_100KF_MASK |
944afa08fdeSSky Huang MRVL_TR_FIX_1000KP_MASK | MRVL_TR_FIX_1000KF_MASK,
945afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_100KP_MASK, 0x6) |
946afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_100KF_MASK, 0x7) |
947afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_1000KP_MASK, 0x6) |
948afa08fdeSSky Huang FIELD_PREP(MRVL_TR_FIX_1000KF_MASK, 0x7));
949ff63cc2eSDaniel Golle
950afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x0, 0xf, 0x3c,
951afa08fdeSSky Huang REMOTE_ACK_COUNT_LIMIT_CTRL_MASK,
952afa08fdeSSky Huang FIELD_PREP(REMOTE_ACK_COUNT_LIMIT_CTRL_MASK, 0x1));
95398c485eaSDaniel Golle phy_restore_page(phydev, MTK_PHY_PAGE_STANDARD, 0);
95498c485eaSDaniel Golle
955ff63cc2eSDaniel Golle /* TR_OPEN_LOOP_EN = 1, lpf_x_average = 10 */
956ff63cc2eSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG234,
957b0f90a86SSkyLake.Huang MTK_PHY_TR_OPEN_LOOP_EN_MASK |
958b0f90a86SSkyLake.Huang MTK_PHY_LPF_X_AVERAGE_MASK,
959ff63cc2eSDaniel Golle BIT(0) | FIELD_PREP(MTK_PHY_LPF_X_AVERAGE_MASK, 0xa));
960ff63cc2eSDaniel Golle
961ff63cc2eSDaniel Golle /* rg_tr_lpf_cnt_val = 1023 */
962ff63cc2eSDaniel Golle phy_write_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_LPF_CNT_VAL, 0x3ff);
96398c485eaSDaniel Golle }
96498c485eaSDaniel Golle
mt798x_phy_eee(struct phy_device * phydev)96598c485eaSDaniel Golle static void mt798x_phy_eee(struct phy_device *phydev)
96698c485eaSDaniel Golle {
96798c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1,
96898c485eaSDaniel Golle MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG120,
96998c485eaSDaniel Golle MTK_PHY_LPI_SIG_EN_LO_THRESH1000_MASK |
97098c485eaSDaniel Golle MTK_PHY_LPI_SIG_EN_HI_THRESH1000_MASK,
97198c485eaSDaniel Golle FIELD_PREP(MTK_PHY_LPI_SIG_EN_LO_THRESH1000_MASK, 0x0) |
97298c485eaSDaniel Golle FIELD_PREP(MTK_PHY_LPI_SIG_EN_HI_THRESH1000_MASK, 0x14));
97398c485eaSDaniel Golle
97498c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1,
97598c485eaSDaniel Golle MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG122,
97698c485eaSDaniel Golle MTK_PHY_LPI_NORM_MSE_HI_THRESH1000_MASK,
97798c485eaSDaniel Golle FIELD_PREP(MTK_PHY_LPI_NORM_MSE_HI_THRESH1000_MASK,
97898c485eaSDaniel Golle 0xff));
97998c485eaSDaniel Golle
98098c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
98198c485eaSDaniel Golle MTK_PHY_RG_TESTMUX_ADC_CTRL,
98298c485eaSDaniel Golle MTK_PHY_RG_TXEN_DIG_MASK);
98398c485eaSDaniel Golle
98498c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1,
98598c485eaSDaniel Golle MTK_PHY_RG_DEV1E_REG19b, MTK_PHY_BYPASS_DSP_LPI_READY);
98698c485eaSDaniel Golle
98798c485eaSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1,
98898c485eaSDaniel Golle MTK_PHY_RG_DEV1E_REG234, MTK_PHY_TR_LP_IIR_EEE_EN);
98998c485eaSDaniel Golle
99098c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG238,
99198c485eaSDaniel Golle MTK_PHY_LPI_SLV_SEND_TX_TIMER_MASK |
99298c485eaSDaniel Golle MTK_PHY_LPI_SLV_SEND_TX_EN,
99398c485eaSDaniel Golle FIELD_PREP(MTK_PHY_LPI_SLV_SEND_TX_TIMER_MASK, 0x120));
99498c485eaSDaniel Golle
995ff63cc2eSDaniel Golle /* Keep MTK_PHY_LPI_SEND_LOC_TIMER as 375 */
996ff63cc2eSDaniel Golle phy_clear_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG239,
997ff63cc2eSDaniel Golle MTK_PHY_LPI_TXPCS_LOC_RCV);
99898c485eaSDaniel Golle
999ff63cc2eSDaniel Golle /* This also fixes some IoT issues, such as CH340 */
100098c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG2C7,
100198c485eaSDaniel Golle MTK_PHY_MAX_GAIN_MASK | MTK_PHY_MIN_GAIN_MASK,
100298c485eaSDaniel Golle FIELD_PREP(MTK_PHY_MAX_GAIN_MASK, 0x8) |
100398c485eaSDaniel Golle FIELD_PREP(MTK_PHY_MIN_GAIN_MASK, 0x13));
100498c485eaSDaniel Golle
100598c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG2D1,
100698c485eaSDaniel Golle MTK_PHY_VCO_SLICER_THRESH_BITS_HIGH_EEE_MASK,
100798c485eaSDaniel Golle FIELD_PREP(MTK_PHY_VCO_SLICER_THRESH_BITS_HIGH_EEE_MASK,
100898c485eaSDaniel Golle 0x33) |
100998c485eaSDaniel Golle MTK_PHY_LPI_SKIP_SD_SLV_TR | MTK_PHY_LPI_TR_READY |
101098c485eaSDaniel Golle MTK_PHY_LPI_VCO_EEE_STG0_EN);
101198c485eaSDaniel Golle
101298c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG323,
101398c485eaSDaniel Golle MTK_PHY_EEE_WAKE_MAS_INT_DC |
101498c485eaSDaniel Golle MTK_PHY_EEE_WAKE_SLV_INT_DC);
101598c485eaSDaniel Golle
101698c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG324,
101798c485eaSDaniel Golle MTK_PHY_SMI_DETCNT_MAX_MASK,
101898c485eaSDaniel Golle FIELD_PREP(MTK_PHY_SMI_DETCNT_MAX_MASK, 0x3f) |
101998c485eaSDaniel Golle MTK_PHY_SMI_DET_MAX_EN);
102098c485eaSDaniel Golle
102198c485eaSDaniel Golle phy_set_bits_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RG_DEV1E_REG326,
102298c485eaSDaniel Golle MTK_PHY_LPI_MODE_SD_ON | MTK_PHY_RESET_RANDUPD_CNT |
102398c485eaSDaniel Golle MTK_PHY_TREC_UPDATE_ENAB_CLR |
102498c485eaSDaniel Golle MTK_PHY_LPI_QUIT_WAIT_DFE_SIG_DET_OFF |
102598c485eaSDaniel Golle MTK_PHY_TR_READY_SKIP_AFE_WAKEUP);
102698c485eaSDaniel Golle
102798c485eaSDaniel Golle phy_select_page(phydev, MTK_PHY_PAGE_EXTENDED_52B5);
10284786eff2SSky Huang __mtk_tr_clr_bits(phydev, 0x2, 0xd, 0x8,
10294786eff2SSky Huang EEE1000_SELECT_SIGNAL_DETECTION_FROM_DFE);
103098c485eaSDaniel Golle
1031afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x2, 0xd, 0xd,
1032afa08fdeSSky Huang EEE1000_STAGE2_TR_KF_MASK,
1033afa08fdeSSky Huang FIELD_PREP(EEE1000_STAGE2_TR_KF_MASK, 0x2));
103498c485eaSDaniel Golle
1035afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x2, 0xd, 0xf,
1036afa08fdeSSky Huang SLAVE_WAKETR_TIMER_MASK | SLAVE_REMTX_TIMER_MASK,
1037afa08fdeSSky Huang FIELD_PREP(SLAVE_WAKETR_TIMER_MASK, 0x6) |
1038afa08fdeSSky Huang FIELD_PREP(SLAVE_REMTX_TIMER_MASK, 0x14));
103998c485eaSDaniel Golle
1040afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x2, 0xd, 0x10,
1041afa08fdeSSky Huang SLAVE_WAKEINT_TIMER_MASK,
1042afa08fdeSSky Huang FIELD_PREP(SLAVE_WAKEINT_TIMER_MASK, 0x8));
104398c485eaSDaniel Golle
1044afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x2, 0xd, 0x14,
1045afa08fdeSSky Huang TR_FREEZE_TIMER2_MASK,
1046afa08fdeSSky Huang FIELD_PREP(TR_FREEZE_TIMER2_MASK, 0x24a));
104798c485eaSDaniel Golle
1048afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x2, 0xd, 0x1c,
1049afa08fdeSSky Huang EEE100_LPSYNC_STAGE1_UPDATE_TIMER_MASK,
1050afa08fdeSSky Huang FIELD_PREP(EEE100_LPSYNC_STAGE1_UPDATE_TIMER_MASK,
1051afa08fdeSSky Huang 0x10));
105298c485eaSDaniel Golle
10534786eff2SSky Huang __mtk_tr_clr_bits(phydev, 0x2, 0xd, 0x25,
10544786eff2SSky Huang WAKE_SLAVE_TR_WAIT_DFE_DETECTION_EN);
105598c485eaSDaniel Golle
1056afa08fdeSSky Huang __mtk_tr_modify(phydev, 0x1, 0xf, 0x0,
1057afa08fdeSSky Huang DFE_TAIL_EANBLE_VGA_TRHESH_1000,
1058afa08fdeSSky Huang FIELD_PREP(DFE_TAIL_EANBLE_VGA_TRHESH_1000, 0x1b));
105998c485eaSDaniel Golle phy_restore_page(phydev, MTK_PHY_PAGE_STANDARD, 0);
106098c485eaSDaniel Golle
106198c485eaSDaniel Golle phy_select_page(phydev, MTK_PHY_PAGE_EXTENDED_3);
1062b0f90a86SSkyLake.Huang __phy_modify(phydev, MTK_PHY_LPI_REG_14,
1063b0f90a86SSkyLake.Huang MTK_PHY_LPI_WAKE_TIMER_1000_MASK,
106498c485eaSDaniel Golle FIELD_PREP(MTK_PHY_LPI_WAKE_TIMER_1000_MASK, 0x19c));
106598c485eaSDaniel Golle
106698c485eaSDaniel Golle __phy_modify(phydev, MTK_PHY_LPI_REG_1c, MTK_PHY_SMI_DET_ON_THRESH_MASK,
106798c485eaSDaniel Golle FIELD_PREP(MTK_PHY_SMI_DET_ON_THRESH_MASK, 0xc));
106898c485eaSDaniel Golle phy_restore_page(phydev, MTK_PHY_PAGE_STANDARD, 0);
106998c485eaSDaniel Golle
107098c485eaSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1,
107198c485eaSDaniel Golle MTK_PHY_RG_LPI_PCS_DSP_CTRL_REG122,
107298c485eaSDaniel Golle MTK_PHY_LPI_NORM_MSE_HI_THRESH1000_MASK,
1073b0f90a86SSkyLake.Huang FIELD_PREP(MTK_PHY_LPI_NORM_MSE_HI_THRESH1000_MASK,
1074b0f90a86SSkyLake.Huang 0xff));
107598c485eaSDaniel Golle }
107698c485eaSDaniel Golle
cal_sw(struct phy_device * phydev,enum CAL_ITEM cal_item,u8 start_pair,u8 end_pair)107798c485eaSDaniel Golle static int cal_sw(struct phy_device *phydev, enum CAL_ITEM cal_item,
107898c485eaSDaniel Golle u8 start_pair, u8 end_pair)
107998c485eaSDaniel Golle {
108098c485eaSDaniel Golle u8 pair_n;
108198c485eaSDaniel Golle int ret;
108298c485eaSDaniel Golle
108398c485eaSDaniel Golle for (pair_n = start_pair; pair_n <= end_pair; pair_n++) {
108498c485eaSDaniel Golle /* TX_OFFSET & TX_AMP have no SW calibration. */
108598c485eaSDaniel Golle switch (cal_item) {
108698c485eaSDaniel Golle case TX_VCM:
108798c485eaSDaniel Golle ret = tx_vcm_cal_sw(phydev, pair_n);
108898c485eaSDaniel Golle break;
108998c485eaSDaniel Golle default:
109098c485eaSDaniel Golle return -EINVAL;
109198c485eaSDaniel Golle }
109298c485eaSDaniel Golle if (ret)
109398c485eaSDaniel Golle return ret;
109498c485eaSDaniel Golle }
109598c485eaSDaniel Golle return 0;
109698c485eaSDaniel Golle }
109798c485eaSDaniel Golle
cal_efuse(struct phy_device * phydev,enum CAL_ITEM cal_item,u8 start_pair,u8 end_pair,u32 * buf)109898c485eaSDaniel Golle static int cal_efuse(struct phy_device *phydev, enum CAL_ITEM cal_item,
109998c485eaSDaniel Golle u8 start_pair, u8 end_pair, u32 *buf)
110098c485eaSDaniel Golle {
110198c485eaSDaniel Golle u8 pair_n;
110298c485eaSDaniel Golle int ret;
110398c485eaSDaniel Golle
110498c485eaSDaniel Golle for (pair_n = start_pair; pair_n <= end_pair; pair_n++) {
110598c485eaSDaniel Golle /* TX_VCM has no efuse calibration. */
110698c485eaSDaniel Golle switch (cal_item) {
110798c485eaSDaniel Golle case REXT:
110898c485eaSDaniel Golle ret = rext_cal_efuse(phydev, buf);
110998c485eaSDaniel Golle break;
111098c485eaSDaniel Golle case TX_OFFSET:
111198c485eaSDaniel Golle ret = tx_offset_cal_efuse(phydev, buf);
111298c485eaSDaniel Golle break;
111398c485eaSDaniel Golle case TX_AMP:
111498c485eaSDaniel Golle ret = tx_amp_cal_efuse(phydev, buf);
111598c485eaSDaniel Golle break;
111698c485eaSDaniel Golle case TX_R50:
111798c485eaSDaniel Golle ret = tx_r50_cal_efuse(phydev, buf, pair_n);
111898c485eaSDaniel Golle break;
111998c485eaSDaniel Golle default:
112098c485eaSDaniel Golle return -EINVAL;
112198c485eaSDaniel Golle }
112298c485eaSDaniel Golle if (ret)
112398c485eaSDaniel Golle return ret;
112498c485eaSDaniel Golle }
112598c485eaSDaniel Golle
112698c485eaSDaniel Golle return 0;
112798c485eaSDaniel Golle }
112898c485eaSDaniel Golle
start_cal(struct phy_device * phydev,enum CAL_ITEM cal_item,enum CAL_MODE cal_mode,u8 start_pair,u8 end_pair,u32 * buf)112998c485eaSDaniel Golle static int start_cal(struct phy_device *phydev, enum CAL_ITEM cal_item,
113098c485eaSDaniel Golle enum CAL_MODE cal_mode, u8 start_pair,
113198c485eaSDaniel Golle u8 end_pair, u32 *buf)
113298c485eaSDaniel Golle {
113398c485eaSDaniel Golle int ret;
113498c485eaSDaniel Golle
113598c485eaSDaniel Golle switch (cal_mode) {
113698c485eaSDaniel Golle case EFUSE_M:
113798c485eaSDaniel Golle ret = cal_efuse(phydev, cal_item, start_pair,
113898c485eaSDaniel Golle end_pair, buf);
113998c485eaSDaniel Golle break;
114098c485eaSDaniel Golle case SW_M:
114198c485eaSDaniel Golle ret = cal_sw(phydev, cal_item, start_pair, end_pair);
114298c485eaSDaniel Golle break;
114398c485eaSDaniel Golle default:
114498c485eaSDaniel Golle return -EINVAL;
114598c485eaSDaniel Golle }
114698c485eaSDaniel Golle
114798c485eaSDaniel Golle if (ret) {
114898c485eaSDaniel Golle phydev_err(phydev, "cal %d failed\n", cal_item);
114998c485eaSDaniel Golle return -EIO;
115098c485eaSDaniel Golle }
115198c485eaSDaniel Golle
115298c485eaSDaniel Golle return 0;
115398c485eaSDaniel Golle }
115498c485eaSDaniel Golle
mt798x_phy_calibration(struct phy_device * phydev)115598c485eaSDaniel Golle static int mt798x_phy_calibration(struct phy_device *phydev)
115698c485eaSDaniel Golle {
115798c485eaSDaniel Golle struct nvmem_cell *cell;
1158b544223bSSkyLake.Huang int ret = 0;
1159b544223bSSkyLake.Huang size_t len;
1160b544223bSSkyLake.Huang u32 *buf;
116198c485eaSDaniel Golle
116298c485eaSDaniel Golle cell = nvmem_cell_get(&phydev->mdio.dev, "phy-cal-data");
116398c485eaSDaniel Golle if (IS_ERR(cell)) {
116498c485eaSDaniel Golle if (PTR_ERR(cell) == -EPROBE_DEFER)
116598c485eaSDaniel Golle return PTR_ERR(cell);
116698c485eaSDaniel Golle return 0;
116798c485eaSDaniel Golle }
116898c485eaSDaniel Golle
116998c485eaSDaniel Golle buf = (u32 *)nvmem_cell_read(cell, &len);
117098c485eaSDaniel Golle if (IS_ERR(buf))
117198c485eaSDaniel Golle return PTR_ERR(buf);
117298c485eaSDaniel Golle nvmem_cell_put(cell);
117398c485eaSDaniel Golle
117498c485eaSDaniel Golle if (!buf[0] || !buf[1] || !buf[2] || !buf[3] || len < 4 * sizeof(u32)) {
117598c485eaSDaniel Golle phydev_err(phydev, "invalid efuse data\n");
117698c485eaSDaniel Golle ret = -EINVAL;
117798c485eaSDaniel Golle goto out;
117898c485eaSDaniel Golle }
117998c485eaSDaniel Golle
118098c485eaSDaniel Golle ret = start_cal(phydev, REXT, EFUSE_M, NO_PAIR, NO_PAIR, buf);
118198c485eaSDaniel Golle if (ret)
118298c485eaSDaniel Golle goto out;
118398c485eaSDaniel Golle ret = start_cal(phydev, TX_OFFSET, EFUSE_M, NO_PAIR, NO_PAIR, buf);
118498c485eaSDaniel Golle if (ret)
118598c485eaSDaniel Golle goto out;
118698c485eaSDaniel Golle ret = start_cal(phydev, TX_AMP, EFUSE_M, NO_PAIR, NO_PAIR, buf);
118798c485eaSDaniel Golle if (ret)
118898c485eaSDaniel Golle goto out;
118998c485eaSDaniel Golle ret = start_cal(phydev, TX_R50, EFUSE_M, PAIR_A, PAIR_D, buf);
119098c485eaSDaniel Golle if (ret)
119198c485eaSDaniel Golle goto out;
119298c485eaSDaniel Golle ret = start_cal(phydev, TX_VCM, SW_M, PAIR_A, PAIR_A, buf);
119398c485eaSDaniel Golle if (ret)
119498c485eaSDaniel Golle goto out;
119598c485eaSDaniel Golle
119698c485eaSDaniel Golle out:
119798c485eaSDaniel Golle kfree(buf);
119898c485eaSDaniel Golle return ret;
119998c485eaSDaniel Golle }
120098c485eaSDaniel Golle
mt798x_phy_config_init(struct phy_device * phydev)120198c485eaSDaniel Golle static int mt798x_phy_config_init(struct phy_device *phydev)
120298c485eaSDaniel Golle {
120398c485eaSDaniel Golle switch (phydev->drv->phy_id) {
120498c485eaSDaniel Golle case MTK_GPHY_ID_MT7981:
120598c485eaSDaniel Golle mt7981_phy_finetune(phydev);
120698c485eaSDaniel Golle break;
120798c485eaSDaniel Golle case MTK_GPHY_ID_MT7988:
120898c485eaSDaniel Golle mt7988_phy_finetune(phydev);
120998c485eaSDaniel Golle break;
121098c485eaSDaniel Golle }
121198c485eaSDaniel Golle
121298c485eaSDaniel Golle mt798x_phy_common_finetune(phydev);
121398c485eaSDaniel Golle mt798x_phy_eee(phydev);
121498c485eaSDaniel Golle
121598c485eaSDaniel Golle return mt798x_phy_calibration(phydev);
121698c485eaSDaniel Golle }
121798c485eaSDaniel Golle
mt798x_phy_led_blink_set(struct phy_device * phydev,u8 index,unsigned long * delay_on,unsigned long * delay_off)1218c66937b0SDaniel Golle static int mt798x_phy_led_blink_set(struct phy_device *phydev, u8 index,
1219c66937b0SDaniel Golle unsigned long *delay_on,
1220c66937b0SDaniel Golle unsigned long *delay_off)
1221c66937b0SDaniel Golle {
1222c66937b0SDaniel Golle bool blinking = false;
12237f9c320cSSkyLake.Huang int err;
1224c66937b0SDaniel Golle
12257f9c320cSSkyLake.Huang err = mtk_phy_led_num_dly_cfg(index, delay_on, delay_off, &blinking);
12267f9c320cSSkyLake.Huang if (err < 0)
12277f9c320cSSkyLake.Huang return err;
1228c66937b0SDaniel Golle
12297f9c320cSSkyLake.Huang err = mtk_phy_hw_led_blink_set(phydev, index, blinking);
1230c66937b0SDaniel Golle if (err)
1231c66937b0SDaniel Golle return err;
1232c66937b0SDaniel Golle
12337f9c320cSSkyLake.Huang return mtk_phy_hw_led_on_set(phydev, index, MTK_GPHY_LED_ON_MASK,
12347f9c320cSSkyLake.Huang false);
1235c66937b0SDaniel Golle }
1236c66937b0SDaniel Golle
mt798x_phy_led_brightness_set(struct phy_device * phydev,u8 index,enum led_brightness value)1237c66937b0SDaniel Golle static int mt798x_phy_led_brightness_set(struct phy_device *phydev,
1238c66937b0SDaniel Golle u8 index, enum led_brightness value)
1239c66937b0SDaniel Golle {
1240c66937b0SDaniel Golle int err;
1241c66937b0SDaniel Golle
12427f9c320cSSkyLake.Huang err = mtk_phy_hw_led_blink_set(phydev, index, false);
1243c66937b0SDaniel Golle if (err)
1244c66937b0SDaniel Golle return err;
1245c66937b0SDaniel Golle
12467f9c320cSSkyLake.Huang return mtk_phy_hw_led_on_set(phydev, index, MTK_GPHY_LED_ON_MASK,
12477f9c320cSSkyLake.Huang (value != LED_OFF));
1248c66937b0SDaniel Golle }
1249c66937b0SDaniel Golle
1250b544223bSSkyLake.Huang static const unsigned long supported_triggers =
1251b544223bSSkyLake.Huang BIT(TRIGGER_NETDEV_FULL_DUPLEX) |
1252c66937b0SDaniel Golle BIT(TRIGGER_NETDEV_HALF_DUPLEX) |
1253c66937b0SDaniel Golle BIT(TRIGGER_NETDEV_LINK) |
1254c66937b0SDaniel Golle BIT(TRIGGER_NETDEV_LINK_10) |
1255c66937b0SDaniel Golle BIT(TRIGGER_NETDEV_LINK_100) |
1256c66937b0SDaniel Golle BIT(TRIGGER_NETDEV_LINK_1000) |
1257c66937b0SDaniel Golle BIT(TRIGGER_NETDEV_RX) |
1258b544223bSSkyLake.Huang BIT(TRIGGER_NETDEV_TX);
1259c66937b0SDaniel Golle
mt798x_phy_led_hw_is_supported(struct phy_device * phydev,u8 index,unsigned long rules)1260c66937b0SDaniel Golle static int mt798x_phy_led_hw_is_supported(struct phy_device *phydev, u8 index,
1261c66937b0SDaniel Golle unsigned long rules)
1262c66937b0SDaniel Golle {
12637f9c320cSSkyLake.Huang return mtk_phy_led_hw_is_supported(phydev, index, rules,
12647f9c320cSSkyLake.Huang supported_triggers);
12657f9c320cSSkyLake.Huang }
1266c66937b0SDaniel Golle
mt798x_phy_led_hw_control_get(struct phy_device * phydev,u8 index,unsigned long * rules)1267c66937b0SDaniel Golle static int mt798x_phy_led_hw_control_get(struct phy_device *phydev, u8 index,
1268c66937b0SDaniel Golle unsigned long *rules)
1269c66937b0SDaniel Golle {
12707f9c320cSSkyLake.Huang return mtk_phy_led_hw_ctrl_get(phydev, index, rules,
12717f9c320cSSkyLake.Huang MTK_GPHY_LED_ON_SET,
12727f9c320cSSkyLake.Huang MTK_GPHY_LED_RX_BLINK_SET,
12737f9c320cSSkyLake.Huang MTK_GPHY_LED_TX_BLINK_SET);
1274c66937b0SDaniel Golle };
1275c66937b0SDaniel Golle
mt798x_phy_led_hw_control_set(struct phy_device * phydev,u8 index,unsigned long rules)1276c66937b0SDaniel Golle static int mt798x_phy_led_hw_control_set(struct phy_device *phydev, u8 index,
1277c66937b0SDaniel Golle unsigned long rules)
1278c66937b0SDaniel Golle {
12797f9c320cSSkyLake.Huang return mtk_phy_led_hw_ctrl_set(phydev, index, rules,
12807f9c320cSSkyLake.Huang MTK_GPHY_LED_ON_SET,
12817f9c320cSSkyLake.Huang MTK_GPHY_LED_RX_BLINK_SET,
12827f9c320cSSkyLake.Huang MTK_GPHY_LED_TX_BLINK_SET);
1283c66937b0SDaniel Golle };
1284c66937b0SDaniel Golle
mt7988_phy_led_get_polarity(struct phy_device * phydev,int led_num)1285c66937b0SDaniel Golle static bool mt7988_phy_led_get_polarity(struct phy_device *phydev, int led_num)
1286c66937b0SDaniel Golle {
1287dc5a6164SHeiner Kallweit struct mtk_socphy_shared *priv = phy_package_get_priv(phydev);
1288c66937b0SDaniel Golle u32 polarities;
1289c66937b0SDaniel Golle
1290c66937b0SDaniel Golle if (led_num == 0)
1291c66937b0SDaniel Golle polarities = ~(priv->boottrap);
1292c66937b0SDaniel Golle else
1293c66937b0SDaniel Golle polarities = MTK_PHY_LED1_DEFAULT_POLARITIES;
1294c66937b0SDaniel Golle
1295c66937b0SDaniel Golle if (polarities & BIT(phydev->mdio.addr))
1296c66937b0SDaniel Golle return true;
1297c66937b0SDaniel Golle
1298c66937b0SDaniel Golle return false;
1299c66937b0SDaniel Golle }
1300c66937b0SDaniel Golle
mt7988_phy_fix_leds_polarities(struct phy_device * phydev)1301c66937b0SDaniel Golle static int mt7988_phy_fix_leds_polarities(struct phy_device *phydev)
1302c66937b0SDaniel Golle {
1303c66937b0SDaniel Golle struct pinctrl *pinctrl;
1304c66937b0SDaniel Golle int index;
1305c66937b0SDaniel Golle
1306c66937b0SDaniel Golle /* Setup LED polarity according to bootstrap use of LED pins */
1307c66937b0SDaniel Golle for (index = 0; index < 2; ++index)
1308c66937b0SDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND2, index ?
1309c66937b0SDaniel Golle MTK_PHY_LED1_ON_CTRL : MTK_PHY_LED0_ON_CTRL,
1310c66937b0SDaniel Golle MTK_PHY_LED_ON_POLARITY,
1311c66937b0SDaniel Golle mt7988_phy_led_get_polarity(phydev, index) ?
1312c66937b0SDaniel Golle MTK_PHY_LED_ON_POLARITY : 0);
1313c66937b0SDaniel Golle
1314c66937b0SDaniel Golle /* Only now setup pinctrl to avoid bogus blinking */
1315c66937b0SDaniel Golle pinctrl = devm_pinctrl_get_select(&phydev->mdio.dev, "gbe-led");
1316c66937b0SDaniel Golle if (IS_ERR(pinctrl))
1317b0f90a86SSkyLake.Huang dev_err(&phydev->mdio.bus->dev,
1318b0f90a86SSkyLake.Huang "Failed to setup PHY LED pinctrl\n");
1319c66937b0SDaniel Golle
1320c66937b0SDaniel Golle return 0;
1321c66937b0SDaniel Golle }
1322c66937b0SDaniel Golle
mt7988_phy_probe_shared(struct phy_device * phydev)1323c66937b0SDaniel Golle static int mt7988_phy_probe_shared(struct phy_device *phydev)
1324c66937b0SDaniel Golle {
1325c66937b0SDaniel Golle struct device_node *np = dev_of_node(&phydev->mdio.bus->dev);
1326dc5a6164SHeiner Kallweit struct mtk_socphy_shared *shared = phy_package_get_priv(phydev);
132715d7b3dfSFrank Wunderlich struct device_node *pio_np;
1328c66937b0SDaniel Golle struct regmap *regmap;
1329c66937b0SDaniel Golle u32 reg;
1330c66937b0SDaniel Golle int ret;
1331c66937b0SDaniel Golle
1332c66937b0SDaniel Golle /* The LED0 of the 4 PHYs in MT7988 are wired to SoC pins LED_A, LED_B,
1333c66937b0SDaniel Golle * LED_C and LED_D respectively. At the same time those pins are used to
1334c66937b0SDaniel Golle * bootstrap configuration of the reference clock source (LED_A),
1335c66937b0SDaniel Golle * DRAM DDRx16b x2/x1 (LED_B) and boot device (LED_C, LED_D).
1336b544223bSSkyLake.Huang * In practice this is done using a LED and a resistor pulling the pin
1337c66937b0SDaniel Golle * either to GND or to VIO.
1338c66937b0SDaniel Golle * The detected value at boot time is accessible at run-time using the
1339c66937b0SDaniel Golle * TPBANK0 register located in the gpio base of the pinctrl, in order
1340c66937b0SDaniel Golle * to read it here it needs to be referenced by a phandle called
1341c66937b0SDaniel Golle * 'mediatek,pio' in the MDIO bus hosting the PHY.
1342c66937b0SDaniel Golle * The 4 bits in TPBANK0 are kept as package shared data and are used to
1343c66937b0SDaniel Golle * set LED polarity for each of the LED0.
1344c66937b0SDaniel Golle */
134515d7b3dfSFrank Wunderlich pio_np = of_parse_phandle(np, "mediatek,pio", 0);
134615d7b3dfSFrank Wunderlich if (!pio_np)
134715d7b3dfSFrank Wunderlich return -ENODEV;
134815d7b3dfSFrank Wunderlich
134915d7b3dfSFrank Wunderlich regmap = device_node_to_regmap(pio_np);
135015d7b3dfSFrank Wunderlich of_node_put(pio_np);
135115d7b3dfSFrank Wunderlich
1352c66937b0SDaniel Golle if (IS_ERR(regmap))
1353c66937b0SDaniel Golle return PTR_ERR(regmap);
1354c66937b0SDaniel Golle
1355c66937b0SDaniel Golle ret = regmap_read(regmap, RG_GPIO_MISC_TPBANK0, ®);
1356c66937b0SDaniel Golle if (ret)
1357c66937b0SDaniel Golle return ret;
1358c66937b0SDaniel Golle
1359c66937b0SDaniel Golle shared->boottrap = FIELD_GET(RG_GPIO_MISC_TPBANK0_BOOTMODE, reg);
1360c66937b0SDaniel Golle
1361c66937b0SDaniel Golle return 0;
1362c66937b0SDaniel Golle }
1363c66937b0SDaniel Golle
mt7988_phy_probe(struct phy_device * phydev)1364c66937b0SDaniel Golle static int mt7988_phy_probe(struct phy_device *phydev)
1365c66937b0SDaniel Golle {
1366c66937b0SDaniel Golle struct mtk_socphy_shared *shared;
1367c66937b0SDaniel Golle struct mtk_socphy_priv *priv;
1368c66937b0SDaniel Golle int err;
1369c66937b0SDaniel Golle
1370c66937b0SDaniel Golle if (phydev->mdio.addr > 3)
1371c66937b0SDaniel Golle return -EINVAL;
1372c66937b0SDaniel Golle
1373c66937b0SDaniel Golle err = devm_phy_package_join(&phydev->mdio.dev, phydev, 0,
1374c66937b0SDaniel Golle sizeof(struct mtk_socphy_shared));
1375c66937b0SDaniel Golle if (err)
1376c66937b0SDaniel Golle return err;
1377c66937b0SDaniel Golle
1378c66937b0SDaniel Golle if (phy_package_probe_once(phydev)) {
1379c66937b0SDaniel Golle err = mt7988_phy_probe_shared(phydev);
1380c66937b0SDaniel Golle if (err)
1381c66937b0SDaniel Golle return err;
1382c66937b0SDaniel Golle }
1383c66937b0SDaniel Golle
1384dc5a6164SHeiner Kallweit shared = phy_package_get_priv(phydev);
1385c66937b0SDaniel Golle priv = &shared->priv[phydev->mdio.addr];
1386c66937b0SDaniel Golle
1387c66937b0SDaniel Golle phydev->priv = priv;
1388c66937b0SDaniel Golle
13897f9c320cSSkyLake.Huang mtk_phy_leds_state_init(phydev);
1390c66937b0SDaniel Golle
1391c66937b0SDaniel Golle err = mt7988_phy_fix_leds_polarities(phydev);
1392c66937b0SDaniel Golle if (err)
1393c66937b0SDaniel Golle return err;
1394c66937b0SDaniel Golle
1395ff63cc2eSDaniel Golle /* Disable TX power saving at probing to:
1396ff63cc2eSDaniel Golle * 1. Meet common mode compliance test criteria
1397ff63cc2eSDaniel Golle * 2. Make sure that TX-VCM calibration works fine
1398ff63cc2eSDaniel Golle */
1399ff63cc2eSDaniel Golle phy_modify_mmd(phydev, MDIO_MMD_VEND1, MTK_PHY_RXADC_CTRL_RG7,
1400ff63cc2eSDaniel Golle MTK_PHY_DA_AD_BUF_BIAS_LP_MASK, 0x3 << 8);
1401ff63cc2eSDaniel Golle
1402c66937b0SDaniel Golle return mt798x_phy_calibration(phydev);
1403c66937b0SDaniel Golle }
1404c66937b0SDaniel Golle
mt7981_phy_probe(struct phy_device * phydev)1405c66937b0SDaniel Golle static int mt7981_phy_probe(struct phy_device *phydev)
1406c66937b0SDaniel Golle {
1407c66937b0SDaniel Golle struct mtk_socphy_priv *priv;
1408c66937b0SDaniel Golle
1409c66937b0SDaniel Golle priv = devm_kzalloc(&phydev->mdio.dev, sizeof(struct mtk_socphy_priv),
1410c66937b0SDaniel Golle GFP_KERNEL);
1411c66937b0SDaniel Golle if (!priv)
1412c66937b0SDaniel Golle return -ENOMEM;
1413c66937b0SDaniel Golle
1414c66937b0SDaniel Golle phydev->priv = priv;
1415c66937b0SDaniel Golle
14167f9c320cSSkyLake.Huang mtk_phy_leds_state_init(phydev);
1417c66937b0SDaniel Golle
1418c66937b0SDaniel Golle return mt798x_phy_calibration(phydev);
1419c66937b0SDaniel Golle }
1420c66937b0SDaniel Golle
an7581_phy_probe(struct phy_device * phydev)14216a325aedSChristian Marangi static int an7581_phy_probe(struct phy_device *phydev)
14226a325aedSChristian Marangi {
14236a325aedSChristian Marangi struct mtk_socphy_priv *priv;
14246a325aedSChristian Marangi struct pinctrl *pinctrl;
14256a325aedSChristian Marangi
14266a325aedSChristian Marangi /* Toggle pinctrl to enable PHY LED */
14276a325aedSChristian Marangi pinctrl = devm_pinctrl_get_select(&phydev->mdio.dev, "gbe-led");
14286a325aedSChristian Marangi if (IS_ERR(pinctrl))
14296a325aedSChristian Marangi dev_err(&phydev->mdio.bus->dev,
14306a325aedSChristian Marangi "Failed to setup PHY LED pinctrl\n");
14316a325aedSChristian Marangi
14326a325aedSChristian Marangi priv = devm_kzalloc(&phydev->mdio.dev, sizeof(*priv), GFP_KERNEL);
14336a325aedSChristian Marangi if (!priv)
14346a325aedSChristian Marangi return -ENOMEM;
14356a325aedSChristian Marangi
14366a325aedSChristian Marangi phydev->priv = priv;
14376a325aedSChristian Marangi
14386a325aedSChristian Marangi return 0;
14396a325aedSChristian Marangi }
14406a325aedSChristian Marangi
an7581_phy_led_polarity_set(struct phy_device * phydev,int index,unsigned long modes)14416a325aedSChristian Marangi static int an7581_phy_led_polarity_set(struct phy_device *phydev, int index,
14426a325aedSChristian Marangi unsigned long modes)
14436a325aedSChristian Marangi {
144400868d03SChristian Marangi u16 val = 0;
14456a325aedSChristian Marangi u32 mode;
14466a325aedSChristian Marangi
14476a325aedSChristian Marangi if (index >= MTK_PHY_MAX_LEDS)
14486a325aedSChristian Marangi return -EINVAL;
14496a325aedSChristian Marangi
14506a325aedSChristian Marangi for_each_set_bit(mode, &modes, __PHY_LED_MODES_NUM) {
14516a325aedSChristian Marangi switch (mode) {
14526a325aedSChristian Marangi case PHY_LED_ACTIVE_LOW:
14536a325aedSChristian Marangi val = MTK_PHY_LED_ON_POLARITY;
14546a325aedSChristian Marangi break;
14556a325aedSChristian Marangi case PHY_LED_ACTIVE_HIGH:
14566a325aedSChristian Marangi break;
14576a325aedSChristian Marangi default:
14586a325aedSChristian Marangi return -EINVAL;
14596a325aedSChristian Marangi }
14606a325aedSChristian Marangi }
14616a325aedSChristian Marangi
14626a325aedSChristian Marangi return phy_modify_mmd(phydev, MDIO_MMD_VEND2, index ?
14636a325aedSChristian Marangi MTK_PHY_LED1_ON_CTRL : MTK_PHY_LED0_ON_CTRL,
14646a325aedSChristian Marangi MTK_PHY_LED_ON_POLARITY, val);
14656a325aedSChristian Marangi }
14666a325aedSChristian Marangi
an7583_phy_config_init(struct phy_device * phydev)1467*8bc3c234SChristian Marangi static int an7583_phy_config_init(struct phy_device *phydev)
1468*8bc3c234SChristian Marangi {
1469*8bc3c234SChristian Marangi /* BMCR_PDOWN is enabled by default */
1470*8bc3c234SChristian Marangi return phy_clear_bits(phydev, MII_BMCR, BMCR_PDOWN);
1471*8bc3c234SChristian Marangi }
1472*8bc3c234SChristian Marangi
147398c485eaSDaniel Golle static struct phy_driver mtk_socphy_driver[] = {
147498c485eaSDaniel Golle {
147598c485eaSDaniel Golle PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7981),
147698c485eaSDaniel Golle .name = "MediaTek MT7981 PHY",
147798c485eaSDaniel Golle .config_init = mt798x_phy_config_init,
147898c485eaSDaniel Golle .config_intr = genphy_no_config_intr,
147998c485eaSDaniel Golle .handle_interrupt = genphy_handle_interrupt_no_ack,
1480c66937b0SDaniel Golle .probe = mt7981_phy_probe,
148198c485eaSDaniel Golle .suspend = genphy_suspend,
148298c485eaSDaniel Golle .resume = genphy_resume,
14833cb1a3c9SSkyLake.Huang .read_page = mtk_phy_read_page,
14843cb1a3c9SSkyLake.Huang .write_page = mtk_phy_write_page,
1485c66937b0SDaniel Golle .led_blink_set = mt798x_phy_led_blink_set,
1486c66937b0SDaniel Golle .led_brightness_set = mt798x_phy_led_brightness_set,
1487c66937b0SDaniel Golle .led_hw_is_supported = mt798x_phy_led_hw_is_supported,
1488c66937b0SDaniel Golle .led_hw_control_set = mt798x_phy_led_hw_control_set,
1489c66937b0SDaniel Golle .led_hw_control_get = mt798x_phy_led_hw_control_get,
149098c485eaSDaniel Golle },
149198c485eaSDaniel Golle {
149298c485eaSDaniel Golle PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7988),
149398c485eaSDaniel Golle .name = "MediaTek MT7988 PHY",
149498c485eaSDaniel Golle .config_init = mt798x_phy_config_init,
149598c485eaSDaniel Golle .config_intr = genphy_no_config_intr,
149698c485eaSDaniel Golle .handle_interrupt = genphy_handle_interrupt_no_ack,
1497c66937b0SDaniel Golle .probe = mt7988_phy_probe,
149898c485eaSDaniel Golle .suspend = genphy_suspend,
149998c485eaSDaniel Golle .resume = genphy_resume,
15003cb1a3c9SSkyLake.Huang .read_page = mtk_phy_read_page,
15013cb1a3c9SSkyLake.Huang .write_page = mtk_phy_write_page,
1502c66937b0SDaniel Golle .led_blink_set = mt798x_phy_led_blink_set,
1503c66937b0SDaniel Golle .led_brightness_set = mt798x_phy_led_brightness_set,
1504c66937b0SDaniel Golle .led_hw_is_supported = mt798x_phy_led_hw_is_supported,
1505c66937b0SDaniel Golle .led_hw_control_set = mt798x_phy_led_hw_control_set,
1506c66937b0SDaniel Golle .led_hw_control_get = mt798x_phy_led_hw_control_get,
150798c485eaSDaniel Golle },
15086a325aedSChristian Marangi {
15096a325aedSChristian Marangi PHY_ID_MATCH_EXACT(MTK_GPHY_ID_AN7581),
15106a325aedSChristian Marangi .name = "Airoha AN7581 PHY",
15116a325aedSChristian Marangi .probe = an7581_phy_probe,
15126a325aedSChristian Marangi .led_blink_set = mt798x_phy_led_blink_set,
15136a325aedSChristian Marangi .led_brightness_set = mt798x_phy_led_brightness_set,
15146a325aedSChristian Marangi .led_hw_is_supported = mt798x_phy_led_hw_is_supported,
15156a325aedSChristian Marangi .led_hw_control_set = mt798x_phy_led_hw_control_set,
15166a325aedSChristian Marangi .led_hw_control_get = mt798x_phy_led_hw_control_get,
15176a325aedSChristian Marangi .led_polarity_set = an7581_phy_led_polarity_set,
15186a325aedSChristian Marangi },
1519*8bc3c234SChristian Marangi {
1520*8bc3c234SChristian Marangi PHY_ID_MATCH_EXACT(MTK_GPHY_ID_AN7583),
1521*8bc3c234SChristian Marangi .name = "Airoha AN7583 PHY",
1522*8bc3c234SChristian Marangi .config_init = an7583_phy_config_init,
1523*8bc3c234SChristian Marangi .probe = an7581_phy_probe,
1524*8bc3c234SChristian Marangi .led_blink_set = mt798x_phy_led_blink_set,
1525*8bc3c234SChristian Marangi .led_brightness_set = mt798x_phy_led_brightness_set,
1526*8bc3c234SChristian Marangi .led_hw_is_supported = mt798x_phy_led_hw_is_supported,
1527*8bc3c234SChristian Marangi .led_hw_control_set = mt798x_phy_led_hw_control_set,
1528*8bc3c234SChristian Marangi .led_hw_control_get = mt798x_phy_led_hw_control_get,
1529*8bc3c234SChristian Marangi .led_polarity_set = an7581_phy_led_polarity_set,
1530*8bc3c234SChristian Marangi },
153198c485eaSDaniel Golle };
153298c485eaSDaniel Golle
153398c485eaSDaniel Golle module_phy_driver(mtk_socphy_driver);
153498c485eaSDaniel Golle
1535b01b59a4SChristophe JAILLET static const struct mdio_device_id __maybe_unused mtk_socphy_tbl[] = {
153698c485eaSDaniel Golle { PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7981) },
153798c485eaSDaniel Golle { PHY_ID_MATCH_EXACT(MTK_GPHY_ID_MT7988) },
15386a325aedSChristian Marangi { PHY_ID_MATCH_EXACT(MTK_GPHY_ID_AN7581) },
1539*8bc3c234SChristian Marangi { PHY_ID_MATCH_EXACT(MTK_GPHY_ID_AN7583) },
154098c485eaSDaniel Golle { }
154198c485eaSDaniel Golle };
154298c485eaSDaniel Golle
154398c485eaSDaniel Golle MODULE_DESCRIPTION("MediaTek SoC Gigabit Ethernet PHY driver");
154498c485eaSDaniel Golle MODULE_AUTHOR("Daniel Golle <daniel@makrotopia.org>");
154598c485eaSDaniel Golle MODULE_AUTHOR("SkyLake Huang <SkyLake.Huang@mediatek.com>");
154698c485eaSDaniel Golle MODULE_LICENSE("GPL");
154798c485eaSDaniel Golle
154898c485eaSDaniel Golle MODULE_DEVICE_TABLE(mdio, mtk_socphy_tbl);
1549