1 /* SPDX-License-Identifier: BSD-3-Clause
2 * Copyright(c) 2018-2021 Beijing WangXun Technology Co., Ltd.
5 #include "ngbe_phy_mvl.h"
7 #define MVL_PHY_RST_WAIT_PERIOD 5
9 s32 ngbe_read_phy_reg_mvl(struct ngbe_hw *hw,
10 u32 reg_addr, u32 device_type, u16 *phy_data)
15 reg.device_type = device_type;
18 if (hw->phy.media_type == ngbe_media_type_fiber)
19 ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 1);
21 ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 0);
23 ngbe_mdi_map_register(®, ®22);
25 ngbe_read_phy_reg_mdi(hw, reg22.addr, reg22.device_type, phy_data);
30 s32 ngbe_write_phy_reg_mvl(struct ngbe_hw *hw,
31 u32 reg_addr, u32 device_type, u16 phy_data)
36 reg.device_type = device_type;
39 if (hw->phy.media_type == ngbe_media_type_fiber)
40 ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 1);
42 ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 0);
44 ngbe_mdi_map_register(®, ®22);
46 ngbe_write_phy_reg_mdi(hw, reg22.addr, reg22.device_type, phy_data);
51 s32 ngbe_check_phy_mode_mvl(struct ngbe_hw *hw)
56 phy_mode = ngbe_flash_read_dword(hw, 0xFF010);
57 value = (u8)(phy_mode >> (hw->bus.lan_id * 8));
59 if (MVL_GEN_CTL_MODE(value) == MVL_GEN_CTL_MODE_COPPER) {
60 /* mode select to RGMII-to-copper */
61 hw->phy.type = ngbe_phy_mvl;
62 hw->phy.media_type = ngbe_media_type_copper;
63 hw->mac.link_type = ngbe_link_copper;
64 } else if (MVL_GEN_CTL_MODE(value) == MVL_GEN_CTL_MODE_FIBER) {
65 /* mode select to RGMII-to-sfi */
66 hw->phy.type = ngbe_phy_mvl_sfi;
67 hw->phy.media_type = ngbe_media_type_fiber;
68 hw->mac.link_type = ngbe_link_fiber;
70 DEBUGOUT("marvell 88E1512 mode %x is not supported.", value);
71 return NGBE_ERR_DEVICE_NOT_SUPPORTED;
77 s32 ngbe_init_phy_mvl(struct ngbe_hw *hw)
83 /* enable interrupts, only link status change and an done is allowed */
84 ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 2);
85 ngbe_read_phy_reg_mdi(hw, MVL_RGM_CTL2, 0, &value);
86 value &= ~MVL_RGM_CTL2_TTC;
87 value |= MVL_RGM_CTL2_RTC;
88 ngbe_write_phy_reg_mdi(hw, MVL_RGM_CTL2, 0, value);
90 hw->phy.write_reg(hw, MVL_CTRL, 0, MVL_CTRL_RESET);
91 for (i = 0; i < 15; i++) {
92 ngbe_read_phy_reg_mdi(hw, MVL_CTRL, 0, &value);
93 if (value & MVL_CTRL_RESET)
100 DEBUGOUT("phy reset exceeds maximum waiting period.");
101 return NGBE_ERR_TIMEOUT;
104 ret_val = hw->phy.reset_hw(hw);
108 /* set LED2 to interrupt output and INTn active low */
109 ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 3);
110 ngbe_read_phy_reg_mdi(hw, MVL_LEDTCR, 0, &value);
111 value |= MVL_LEDTCR_INTR_EN;
112 value &= ~(MVL_LEDTCR_INTR_POL);
113 ngbe_write_phy_reg_mdi(hw, MVL_LEDTCR, 0, value);
115 if (hw->phy.type == ngbe_phy_mvl_sfi) {
116 hw->phy.read_reg(hw, MVL_CTRL1, 0, &value);
117 value &= ~MVL_CTRL1_INTR_POL;
118 ngbe_write_phy_reg_mdi(hw, MVL_CTRL1, 0, value);
121 /* enable link status change and AN complete interrupts */
122 value = MVL_INTR_EN_ANC | MVL_INTR_EN_LSC;
123 hw->phy.write_reg(hw, MVL_INTR_EN, 0, value);
125 hw->phy.set_phy_power(hw, false);
130 s32 ngbe_setup_phy_link_mvl(struct ngbe_hw *hw, u32 speed,
131 bool autoneg_wait_to_complete)
137 UNREFERENCED_PARAMETER(autoneg_wait_to_complete);
139 if (hw->led_conf == 0xFFFF) {
141 ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 3);
142 ngbe_read_phy_reg_mdi(hw, MVL_LEDFCR, 0, &value);
143 value &= ~(MVL_LEDFCR_CTL0 | MVL_LEDFCR_CTL1);
144 value |= MVL_LEDFCR_CTL0_CONF | MVL_LEDFCR_CTL1_CONF;
145 ngbe_write_phy_reg_mdi(hw, MVL_LEDFCR, 0, value);
146 ngbe_read_phy_reg_mdi(hw, MVL_LEDPCR, 0, &value);
147 value &= ~(MVL_LEDPCR_CTL0 | MVL_LEDPCR_CTL1);
148 value |= MVL_LEDPCR_CTL0_CONF | MVL_LEDPCR_CTL1_CONF;
149 ngbe_write_phy_reg_mdi(hw, MVL_LEDPCR, 0, value);
152 hw->phy.autoneg_advertised = 0;
154 if (hw->phy.type == ngbe_phy_mvl) {
155 if (!hw->mac.autoneg) {
157 case NGBE_LINK_SPEED_1GB_FULL:
158 value = MVL_CTRL_SPEED_SELECT1;
160 case NGBE_LINK_SPEED_100M_FULL:
161 value = MVL_CTRL_SPEED_SELECT0;
163 case NGBE_LINK_SPEED_10M_FULL:
167 value = MVL_CTRL_SPEED_SELECT0 |
168 MVL_CTRL_SPEED_SELECT1;
169 DEBUGOUT("unknown speed = 0x%x.", speed);
173 value |= MVL_CTRL_DUPLEX | MVL_CTRL_RESET;
174 ngbe_write_phy_reg_mdi(hw, MVL_CTRL, 0, value);
178 if (speed & NGBE_LINK_SPEED_1GB_FULL) {
179 value_r9 |= MVL_PHY_1000BASET_FULL;
180 hw->phy.autoneg_advertised |= NGBE_LINK_SPEED_1GB_FULL;
183 if (speed & NGBE_LINK_SPEED_100M_FULL) {
184 value_r4 |= MVL_PHY_100BASET_FULL;
185 hw->phy.autoneg_advertised |= NGBE_LINK_SPEED_100M_FULL;
188 if (speed & NGBE_LINK_SPEED_10M_FULL) {
189 value_r4 |= MVL_PHY_10BASET_FULL;
190 hw->phy.autoneg_advertised |= NGBE_LINK_SPEED_10M_FULL;
193 hw->phy.read_reg(hw, MVL_ANA, 0, &value);
194 value &= ~(MVL_PHY_100BASET_FULL |
195 MVL_PHY_100BASET_HALF |
196 MVL_PHY_10BASET_FULL |
197 MVL_PHY_10BASET_HALF);
199 hw->phy.write_reg(hw, MVL_ANA, 0, value_r4);
201 hw->phy.read_reg(hw, MVL_PHY_1000BASET, 0, &value);
202 value &= ~(MVL_PHY_1000BASET_FULL |
203 MVL_PHY_1000BASET_HALF);
205 hw->phy.write_reg(hw, MVL_PHY_1000BASET, 0, value_r9);
207 hw->phy.autoneg_advertised |= NGBE_LINK_SPEED_1GB_FULL;
209 hw->phy.read_reg(hw, MVL_ANA, 0, &value);
210 value &= ~(MVL_PHY_1000BASEX_HALF | MVL_PHY_1000BASEX_FULL);
211 value |= MVL_PHY_1000BASEX_FULL;
212 hw->phy.write_reg(hw, MVL_ANA, 0, value);
215 value = MVL_CTRL_RESTART_AN | MVL_CTRL_ANE | MVL_CTRL_RESET;
216 ngbe_write_phy_reg_mdi(hw, MVL_CTRL, 0, value);
219 hw->phy.set_phy_power(hw, true);
221 hw->phy.read_reg(hw, MVL_INTR, 0, &value);
226 s32 ngbe_reset_phy_mvl(struct ngbe_hw *hw)
232 if (hw->phy.type != ngbe_phy_mvl && hw->phy.type != ngbe_phy_mvl_sfi)
233 return NGBE_ERR_PHY_TYPE;
235 /* select page 18 reg 20 */
236 status = ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 18);
238 /* mode select to RGMII-to-copper or RGMII-to-sfi*/
239 if (hw->phy.type == ngbe_phy_mvl)
240 ctrl = MVL_GEN_CTL_MODE_COPPER;
242 ctrl = MVL_GEN_CTL_MODE_FIBER;
243 status = ngbe_write_phy_reg_mdi(hw, MVL_GEN_CTL, 0, ctrl);
245 ctrl |= MVL_GEN_CTL_RESET;
246 status = ngbe_write_phy_reg_mdi(hw, MVL_GEN_CTL, 0, ctrl);
248 for (i = 0; i < MVL_PHY_RST_WAIT_PERIOD; i++) {
249 status = ngbe_read_phy_reg_mdi(hw, MVL_GEN_CTL, 0, &ctrl);
250 if (!(ctrl & MVL_GEN_CTL_RESET))
255 if (i == MVL_PHY_RST_WAIT_PERIOD) {
256 DEBUGOUT("PHY reset polling failed to complete.");
257 return NGBE_ERR_RESET_FAILED;
263 s32 ngbe_get_phy_advertised_pause_mvl(struct ngbe_hw *hw, u8 *pause_bit)
268 if (hw->phy.type == ngbe_phy_mvl) {
269 status = hw->phy.read_reg(hw, MVL_ANA, 0, &value);
270 value &= MVL_CANA_ASM_PAUSE | MVL_CANA_PAUSE;
271 *pause_bit = (u8)(value >> 10);
273 status = hw->phy.read_reg(hw, MVL_ANA, 0, &value);
274 value &= MVL_FANA_PAUSE_MASK;
275 *pause_bit = (u8)(value >> 7);
281 s32 ngbe_get_phy_lp_advertised_pause_mvl(struct ngbe_hw *hw, u8 *pause_bit)
286 if (hw->phy.type == ngbe_phy_mvl) {
287 status = hw->phy.read_reg(hw, MVL_LPAR, 0, &value);
288 value &= MVL_CLPAR_ASM_PAUSE | MVL_CLPAR_PAUSE;
289 *pause_bit = (u8)(value >> 10);
291 status = hw->phy.read_reg(hw, MVL_LPAR, 0, &value);
292 value &= MVL_FLPAR_PAUSE_MASK;
293 *pause_bit = (u8)(value >> 7);
299 s32 ngbe_set_phy_pause_adv_mvl(struct ngbe_hw *hw, u16 pause_bit)
304 if (hw->phy.type == ngbe_phy_mvl) {
305 status = hw->phy.read_reg(hw, MVL_ANA, 0, &value);
306 value &= ~(MVL_CANA_ASM_PAUSE | MVL_CANA_PAUSE);
308 status = hw->phy.read_reg(hw, MVL_ANA, 0, &value);
309 value &= ~MVL_FANA_PAUSE_MASK;
313 status = hw->phy.write_reg(hw, MVL_ANA, 0, value);
318 s32 ngbe_check_phy_link_mvl(struct ngbe_hw *hw,
319 u32 *speed, bool *link_up)
327 /* Initialize speed and link to default case */
329 *speed = NGBE_LINK_SPEED_UNKNOWN;
331 hw->phy.read_reg(hw, MVL_INTR, 0, &insr);
334 * Check current speed and link status of the PHY register.
335 * This is a vendor specific register and may have to
336 * be changed for other copper PHYs.
338 status = hw->phy.read_reg(hw, MVL_PHYSR, 0, &phy_data);
339 phy_link = phy_data & MVL_PHYSR_LINK;
340 phy_speed = phy_data & MVL_PHYSR_SPEED_MASK;
342 if (phy_link == MVL_PHYSR_LINK) {
345 if (phy_speed == MVL_PHYSR_SPEED_1000M)
346 *speed = NGBE_LINK_SPEED_1GB_FULL;
347 else if (phy_speed == MVL_PHYSR_SPEED_100M)
348 *speed = NGBE_LINK_SPEED_100M_FULL;
349 else if (phy_speed == MVL_PHYSR_SPEED_10M)
350 *speed = NGBE_LINK_SPEED_10M_FULL;
356 s32 ngbe_set_phy_power_mvl(struct ngbe_hw *hw, bool on)
360 hw->phy.read_reg(hw, MVL_CTRL, 0, &value);
362 value &= ~MVL_CTRL_PWDN;
364 value |= MVL_CTRL_PWDN;
365 hw->phy.write_reg(hw, MVL_CTRL, 0, value);