Lines Matching +full:pcs +full:- +full:phy
1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * Marvell 88E6352 family SERDES PCS support
15 #include "phy.h"
35 err = mdiodev_c45_read(&mpcs->mdio, MDIO_MMD_PHYXS, regnum); in mv88e639x_read()
46 return mdiodev_c45_write(&mpcs->mdio, MDIO_MMD_PHYXS, regnum, val); in mv88e639x_write()
52 return mdiodev_c45_modify(&mpcs->mdio, MDIO_MMD_PHYXS, regnum, mask, in mv88e639x_modify()
59 return mdiodev_c45_modify_changed(&mpcs->mdio, MDIO_MMD_PHYXS, regnum, in mv88e639x_modify_changed()
73 mpcs->mdio.dev.parent = dev; in mv88e639x_pcs_alloc()
74 mpcs->mdio.bus = bus; in mv88e639x_pcs_alloc()
75 mpcs->mdio.addr = addr; in mv88e639x_pcs_alloc()
77 snprintf(mpcs->name, sizeof(mpcs->name), in mv88e639x_pcs_alloc()
78 "mv88e6xxx-%s-serdes-%d", dev_name(dev), port); in mv88e639x_pcs_alloc()
88 handler = READ_ONCE(mpcs->handle_irq); in mv88e639x_pcs_handle_irq()
103 mpcs->sgmii_pcs.poll = true; in mv88e639x_pcs_setup_irq()
104 mpcs->xg_pcs.poll = true; in mv88e639x_pcs_setup_irq()
108 mpcs->irq = irq; in mv88e639x_pcs_setup_irq()
111 IRQF_ONESHOT, mpcs->name, mpcs); in mv88e639x_pcs_setup_irq()
116 struct mv88e639x_pcs *mpcs = chip->ports[port].pcs_private; in mv88e639x_pcs_teardown()
121 if (mpcs->irq) in mv88e639x_pcs_teardown()
122 free_irq(mpcs->irq, mpcs); in mv88e639x_pcs_teardown()
126 chip->ports[port].pcs_private = NULL; in mv88e639x_pcs_teardown()
129 static struct mv88e639x_pcs *sgmii_pcs_to_mv88e639x_pcs(struct phylink_pcs *pcs) in sgmii_pcs_to_mv88e639x_pcs() argument
131 return container_of(pcs, struct mv88e639x_pcs, sgmii_pcs); in sgmii_pcs_to_mv88e639x_pcs()
145 phylink_pcs_change(&mpcs->sgmii_pcs, in mv88e639x_sgmii_handle_irq()
183 static int mv88e639x_sgmii_pcs_enable(struct phylink_pcs *pcs) in mv88e639x_sgmii_pcs_enable() argument
185 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_enable()
188 mpcs->handle_irq = mv88e639x_sgmii_handle_irq; in mv88e639x_sgmii_pcs_enable()
190 return mv88e639x_sgmii_pcs_control_irq(mpcs, !!mpcs->irq); in mv88e639x_sgmii_pcs_enable()
193 static void mv88e639x_sgmii_pcs_disable(struct phylink_pcs *pcs) in mv88e639x_sgmii_pcs_disable() argument
195 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_disable()
201 static void mv88e639x_sgmii_pcs_pre_config(struct phylink_pcs *pcs, in mv88e639x_sgmii_pcs_pre_config() argument
204 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_pre_config()
224 err = mdiobus_c45_write(mpcs->mdio.bus, lanes[i], in mv88e6390_erratum_3_14()
230 err = mdiobus_c45_write(mpcs->mdio.bus, lanes[i], in mv88e6390_erratum_3_14()
240 static int mv88e639x_sgmii_pcs_post_config(struct phylink_pcs *pcs, in mv88e639x_sgmii_pcs_post_config() argument
243 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_post_config()
248 if (mpcs->erratum_3_14) { in mv88e639x_sgmii_pcs_post_config()
251 dev_err(mpcs->mdio.dev.parent, in mv88e639x_sgmii_pcs_post_config()
259 static void mv88e639x_sgmii_pcs_get_state(struct phylink_pcs *pcs, in mv88e639x_sgmii_pcs_get_state() argument
262 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_get_state()
268 dev_err(mpcs->mdio.dev.parent, in mv88e639x_sgmii_pcs_get_state()
269 "can't read Serdes PHY %s: %pe\n", in mv88e639x_sgmii_pcs_get_state()
271 state->link = false; in mv88e639x_sgmii_pcs_get_state()
277 dev_err(mpcs->mdio.dev.parent, in mv88e639x_sgmii_pcs_get_state()
278 "can't read Serdes PHY %s: %pe\n", in mv88e639x_sgmii_pcs_get_state()
280 state->link = false; in mv88e639x_sgmii_pcs_get_state()
286 dev_err(mpcs->mdio.dev.parent, in mv88e639x_sgmii_pcs_get_state()
287 "can't read Serdes PHY %s: %pe\n", in mv88e639x_sgmii_pcs_get_state()
289 state->link = false; in mv88e639x_sgmii_pcs_get_state()
293 mv88e6xxx_pcs_decode_state(mpcs->mdio.dev.parent, bmsr, lpa, status, in mv88e639x_sgmii_pcs_get_state()
297 static int mv88e639x_sgmii_pcs_config(struct phylink_pcs *pcs, in mv88e639x_sgmii_pcs_config() argument
303 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_config()
312 mpcs->interface = interface; in mv88e639x_sgmii_pcs_config()
337 static void mv88e639x_sgmii_pcs_an_restart(struct phylink_pcs *pcs) in mv88e639x_sgmii_pcs_an_restart() argument
339 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_an_restart()
345 static void mv88e639x_sgmii_pcs_link_up(struct phylink_pcs *pcs, in mv88e639x_sgmii_pcs_link_up() argument
350 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_sgmii_pcs_link_up()
363 dev_err(mpcs->mdio.dev.parent, in mv88e639x_sgmii_pcs_link_up()
364 "can't access Serdes PHY %s: %pe\n", in mv88e639x_sgmii_pcs_link_up()
379 static struct mv88e639x_pcs *xg_pcs_to_mv88e639x_pcs(struct phylink_pcs *pcs) in xg_pcs_to_mv88e639x_pcs() argument
381 return container_of(pcs, struct mv88e639x_pcs, xg_pcs); in xg_pcs_to_mv88e639x_pcs()
397 static void mv88e639x_xg_pcs_get_state(struct phylink_pcs *pcs, in mv88e639x_xg_pcs_get_state() argument
400 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e639x_xg_pcs_get_state()
404 state->link = false; in mv88e639x_xg_pcs_get_state()
408 dev_err(mpcs->mdio.dev.parent, in mv88e639x_xg_pcs_get_state()
409 "can't read Serdes PHY %s: %pe\n", in mv88e639x_xg_pcs_get_state()
414 state->link = !!(status & MDIO_STAT1_LSTATUS); in mv88e639x_xg_pcs_get_state()
415 if (state->link) { in mv88e639x_xg_pcs_get_state()
416 switch (state->interface) { in mv88e639x_xg_pcs_get_state()
418 state->speed = SPEED_5000; in mv88e639x_xg_pcs_get_state()
424 state->speed = SPEED_10000; in mv88e639x_xg_pcs_get_state()
428 state->link = false; in mv88e639x_xg_pcs_get_state()
432 state->duplex = DUPLEX_FULL; in mv88e639x_xg_pcs_get_state()
436 static int mv88e639x_xg_pcs_config(struct phylink_pcs *pcs, in mv88e639x_xg_pcs_config() argument
451 mpcs = chip->ports[port].pcs_private; in mv88e639x_pcs_select()
459 return &mpcs->sgmii_pcs; in mv88e639x_pcs_select()
462 if (!mpcs->supports_5g) in mv88e639x_pcs_select()
469 return &mpcs->xg_pcs; in mv88e639x_pcs_select()
489 phylink_pcs_change(&mpcs->xg_pcs, in mv88e6390_xg_handle_irq()
510 static int mv88e6390_xg_pcs_enable(struct phylink_pcs *pcs) in mv88e6390_xg_pcs_enable() argument
512 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e6390_xg_pcs_enable()
519 mpcs->handle_irq = mv88e6390_xg_handle_irq; in mv88e6390_xg_pcs_enable()
521 return mv88e6390_xg_control_irq(mpcs, !!mpcs->irq); in mv88e6390_xg_pcs_enable()
524 static void mv88e6390_xg_pcs_disable(struct phylink_pcs *pcs) in mv88e6390_xg_pcs_disable() argument
526 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e6390_xg_pcs_disable()
558 dev = chip->dev; in mv88e6390_pcs_init()
562 return -ENOMEM; in mv88e6390_pcs_init()
564 mpcs->sgmii_pcs.ops = &mv88e639x_sgmii_pcs_ops; in mv88e6390_pcs_init()
565 mpcs->sgmii_pcs.neg_mode = true; in mv88e6390_pcs_init()
566 mpcs->xg_pcs.ops = &mv88e6390_xg_pcs_ops; in mv88e6390_pcs_init()
567 mpcs->xg_pcs.neg_mode = true; in mv88e6390_pcs_init()
569 if (chip->info->prod_num == MV88E6XXX_PORT_SWITCH_ID_PROD_6190X || in mv88e6390_pcs_init()
570 chip->info->prod_num == MV88E6XXX_PORT_SWITCH_ID_PROD_6390X) in mv88e6390_pcs_init()
571 mpcs->erratum_3_14 = true; in mv88e6390_pcs_init()
585 chip->ports[port].pcs_private = mpcs; in mv88e6390_pcs_init()
643 * When a SERDES port is operating in 1000BASE-X or SGMII mode link may not
691 err = mdiodev_c45_modify(&mpcs->mdio, fixes[i].dev, in mv88e6393x_erratum_5_2()
701 /* Inband AN is broken on Amethyst in 2500base-x mode when set by standard
703 * We can get around this by configuring the PCS mode to 1000base-x and then
707 * It seem that when we do this configuration to 2500base-x mode (by changing
708 * PCS mode to 1000base-x and frequency to 3.125 GHz from 1.25 GHz) and then
709 * configure to sgmii or 1000base-x, the device thinks that it already has
712 * To avoid this, change PCS mode back to 2500base-x when disabling SerDes from
713 * 2500base-x mode.
735 return mdiodev_c45_write(&mpcs->mdio, MDIO_MMD_VEND1, 0x8000, 0x58); in mv88e6393x_fix_2500basex_an()
749 dev_err(mpcs->mdio.dev.parent, in mv88e6393x_sgmii_apply_2500basex_an()
756 static void mv88e6393x_sgmii_pcs_disable(struct phylink_pcs *pcs) in mv88e6393x_sgmii_pcs_disable() argument
758 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_sgmii_pcs_disable()
760 mv88e639x_sgmii_pcs_disable(pcs); in mv88e6393x_sgmii_pcs_disable()
762 mv88e6393x_sgmii_apply_2500basex_an(mpcs, mpcs->interface, false); in mv88e6393x_sgmii_pcs_disable()
765 static void mv88e6393x_sgmii_pcs_pre_config(struct phylink_pcs *pcs, in mv88e6393x_sgmii_pcs_pre_config() argument
768 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_sgmii_pcs_pre_config()
770 mv88e639x_sgmii_pcs_pre_config(pcs, interface); in mv88e6393x_sgmii_pcs_pre_config()
772 mv88e6393x_sgmii_apply_2500basex_an(mpcs, mpcs->interface, false); in mv88e6393x_sgmii_pcs_pre_config()
775 static int mv88e6393x_sgmii_pcs_post_config(struct phylink_pcs *pcs, in mv88e6393x_sgmii_pcs_post_config() argument
778 struct mv88e639x_pcs *mpcs = sgmii_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_sgmii_pcs_post_config()
793 return mv88e639x_sgmii_pcs_post_config(pcs, interface); in mv88e6393x_sgmii_pcs_post_config()
824 phylink_pcs_change(&mpcs->xg_pcs, !link_down); in mv88e6393x_xg_handle_irq()
843 static int mv88e6393x_xg_pcs_enable(struct phylink_pcs *pcs) in mv88e6393x_xg_pcs_enable() argument
845 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_xg_pcs_enable()
847 mpcs->handle_irq = mv88e6393x_xg_handle_irq; in mv88e6393x_xg_pcs_enable()
849 return mv88e6393x_xg_control_irq(mpcs, !!mpcs->irq); in mv88e6393x_xg_pcs_enable()
852 static void mv88e6393x_xg_pcs_disable(struct phylink_pcs *pcs) in mv88e6393x_xg_pcs_disable() argument
854 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_xg_pcs_disable()
861 /* The PCS has to be powered down while CMODE is changed */
862 static void mv88e6393x_xg_pcs_pre_config(struct phylink_pcs *pcs, in mv88e6393x_xg_pcs_pre_config() argument
865 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_xg_pcs_pre_config()
871 static int mv88e6393x_xg_pcs_post_config(struct phylink_pcs *pcs, in mv88e6393x_xg_pcs_post_config() argument
874 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_xg_pcs_post_config()
891 static void mv88e6393x_xg_pcs_get_state(struct phylink_pcs *pcs, in mv88e6393x_xg_pcs_get_state() argument
894 struct mv88e639x_pcs *mpcs = xg_pcs_to_mv88e639x_pcs(pcs); in mv88e6393x_xg_pcs_get_state()
898 if (state->interface != PHY_INTERFACE_MODE_USXGMII) in mv88e6393x_xg_pcs_get_state()
899 return mv88e639x_xg_pcs_get_state(pcs, state); in mv88e6393x_xg_pcs_get_state()
901 state->link = false; in mv88e6393x_xg_pcs_get_state()
906 dev_err(mpcs->mdio.dev.parent, in mv88e6393x_xg_pcs_get_state()
911 state->link = !!(status & MDIO_USXGMII_LINK); in mv88e6393x_xg_pcs_get_state()
912 state->an_complete = state->link; in mv88e6393x_xg_pcs_get_state()
937 dev = chip->dev; in mv88e6393x_pcs_init()
941 return -ENOMEM; in mv88e6393x_pcs_init()
943 mpcs->sgmii_pcs.ops = &mv88e6393x_sgmii_pcs_ops; in mv88e6393x_pcs_init()
944 mpcs->sgmii_pcs.neg_mode = true; in mv88e6393x_pcs_init()
945 mpcs->xg_pcs.ops = &mv88e6393x_xg_pcs_ops; in mv88e6393x_pcs_init()
946 mpcs->xg_pcs.neg_mode = true; in mv88e6393x_pcs_init()
947 mpcs->supports_5g = true; in mv88e6393x_pcs_init()
957 chip->ports[port].pcs_private = mpcs; in mv88e6393x_pcs_init()