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_setup_phy_link_mvl(struct ngbe_hw *hw, u32 speed,
52 bool autoneg_wait_to_complete)
58 DEBUGFUNC("ngbe_setup_phy_link_mvl");
59 UNREFERENCED_PARAMETER(autoneg_wait_to_complete);
61 hw->phy.autoneg_advertised = 0;
63 if (hw->phy.type == ngbe_phy_mvl) {
64 if (speed & NGBE_LINK_SPEED_1GB_FULL) {
65 value_r9 |= MVL_PHY_1000BASET_FULL;
66 hw->phy.autoneg_advertised |= NGBE_LINK_SPEED_1GB_FULL;
69 if (speed & NGBE_LINK_SPEED_100M_FULL) {
70 value_r4 |= MVL_PHY_100BASET_FULL;
71 hw->phy.autoneg_advertised |= NGBE_LINK_SPEED_100M_FULL;
74 if (speed & NGBE_LINK_SPEED_10M_FULL) {
75 value_r4 |= MVL_PHY_10BASET_FULL;
76 hw->phy.autoneg_advertised |= NGBE_LINK_SPEED_10M_FULL;
79 hw->phy.read_reg(hw, MVL_ANA, 0, &value);
80 value &= ~(MVL_PHY_100BASET_FULL |
81 MVL_PHY_100BASET_HALF |
82 MVL_PHY_10BASET_FULL |
83 MVL_PHY_10BASET_HALF);
85 hw->phy.write_reg(hw, MVL_ANA, 0, value_r4);
87 hw->phy.read_reg(hw, MVL_PHY_1000BASET, 0, &value);
88 value &= ~(MVL_PHY_1000BASET_FULL |
89 MVL_PHY_1000BASET_HALF);
91 hw->phy.write_reg(hw, MVL_PHY_1000BASET, 0, value_r9);
93 hw->phy.autoneg_advertised = 1;
95 hw->phy.read_reg(hw, MVL_ANA, 0, &value);
96 value &= ~(MVL_PHY_1000BASEX_HALF | MVL_PHY_1000BASEX_FULL);
97 value |= MVL_PHY_1000BASEX_FULL;
98 hw->phy.write_reg(hw, MVL_ANA, 0, value);
101 value = MVL_CTRL_RESTART_AN | MVL_CTRL_ANE;
102 ngbe_write_phy_reg_mdi(hw, MVL_CTRL, 0, value);
104 hw->phy.read_reg(hw, MVL_INTR, 0, &value);
109 s32 ngbe_reset_phy_mvl(struct ngbe_hw *hw)
115 DEBUGFUNC("ngbe_reset_phy_mvl");
117 if (hw->phy.type != ngbe_phy_mvl && hw->phy.type != ngbe_phy_mvl_sfi)
118 return NGBE_ERR_PHY_TYPE;
120 /* select page 18 reg 20 */
121 status = ngbe_write_phy_reg_mdi(hw, MVL_PAGE_SEL, 0, 18);
123 /* mode select to RGMII-to-copper or RGMII-to-sfi*/
124 if (hw->phy.type == ngbe_phy_mvl)
125 ctrl = MVL_GEN_CTL_MODE_COPPER;
127 ctrl = MVL_GEN_CTL_MODE_FIBER;
128 status = ngbe_write_phy_reg_mdi(hw, MVL_GEN_CTL, 0, ctrl);
130 ctrl |= MVL_GEN_CTL_RESET;
131 status = ngbe_write_phy_reg_mdi(hw, MVL_GEN_CTL, 0, ctrl);
133 for (i = 0; i < MVL_PHY_RST_WAIT_PERIOD; i++) {
134 status = ngbe_read_phy_reg_mdi(hw, MVL_GEN_CTL, 0, &ctrl);
135 if (!(ctrl & MVL_GEN_CTL_RESET))
140 if (i == MVL_PHY_RST_WAIT_PERIOD) {
141 DEBUGOUT("PHY reset polling failed to complete.\n");
142 return NGBE_ERR_RESET_FAILED;
148 s32 ngbe_check_phy_link_mvl(struct ngbe_hw *hw,
149 u32 *speed, bool *link_up)
157 DEBUGFUNC("ngbe_check_phy_link_mvl");
159 /* Initialize speed and link to default case */
161 *speed = NGBE_LINK_SPEED_UNKNOWN;
163 hw->phy.read_reg(hw, MVL_INTR, 0, &insr);
166 * Check current speed and link status of the PHY register.
167 * This is a vendor specific register and may have to
168 * be changed for other copper PHYs.
170 status = hw->phy.read_reg(hw, MVL_PHYSR, 0, &phy_data);
171 phy_link = phy_data & MVL_PHYSR_LINK;
172 phy_speed = phy_data & MVL_PHYSR_SPEED_MASK;
174 if (phy_link == MVL_PHYSR_LINK) {
177 if (phy_speed == MVL_PHYSR_SPEED_1000M)
178 *speed = NGBE_LINK_SPEED_1GB_FULL;
179 else if (phy_speed == MVL_PHYSR_SPEED_100M)
180 *speed = NGBE_LINK_SPEED_100M_FULL;
181 else if (phy_speed == MVL_PHYSR_SPEED_10M)
182 *speed = NGBE_LINK_SPEED_10M_FULL;