Re: [PATCH net] net: phy: micrel: fix KSZ9477 PHY issues after suspend/resume

From: Robert Hancock
Date: Tue May 28 2024 - 18:51:57 EST


On Tue, 2024-05-28 at 14:37 -0700, Tristram.Ha@xxxxxxxxxxxxx wrote:
> CAUTION: This email originated from outside of the organization. Do
> not click links or open attachments unless you recognize the sender
> and know the content is safe.
>
> From: Tristram Ha <tristram.ha@xxxxxxxxxxxxx>
>
> When the PHY is powered up after powered down most of the registers
> are
> reset, so the PHY setup code needs to be done again.  In addition the
> interrupt register will need to be setup again so that link status
> indication works again.
>
> Fixes: 26dd2974c5b5 ("net: phy: micrel: Move KSZ9477 errata fixes to
> PHY driver")
> Signed-off-by: Tristram Ha <tristram.ha@xxxxxxxxxxxxx>
> ---
>  drivers/net/phy/micrel.c | 60 +++++++++++++++++++++++++++++++++++---
> --
>  1 file changed, 53 insertions(+), 7 deletions(-)
>
> diff --git a/drivers/net/phy/micrel.c b/drivers/net/phy/micrel.c
> index 2b8f8b7f1517..902d9a262c8a 100644
> --- a/drivers/net/phy/micrel.c
> +++ b/drivers/net/phy/micrel.c
> @@ -1918,7 +1918,7 @@ static const struct ksz9477_errata_write
> ksz9477_errata_writes[] = {
>         {0x01, 0x75, 0x0060},
>         {0x01, 0xd3, 0x7777},
>         {0x1c, 0x06, 0x3008},
> -       {0x1c, 0x08, 0x2000},
> +       {0x1c, 0x08, 0x2001},

This change wasn't mentioned in the commit message, but I see that the
latest errata sheet says "The value of this register may read back as
either 0x2000 or 0x2001. Bit 0 is read-only, and is not a fixed
value."

>
>         /* Transmit waveform amplitude can be improved (1000BASE-T,
> 100BASE-TX, 10BASE-Te) */
>         {0x1c, 0x04, 0x00d0},
> @@ -1939,7 +1939,7 @@ static const struct ksz9477_errata_write
> ksz9477_errata_writes[] = {
>         {0x1c, 0x20, 0xeeee},
>  };
>
> -static int ksz9477_config_init(struct phy_device *phydev)
> +static int ksz9477_phy_errata(struct phy_device *phydev)
>  {
>         int err;
>         int i;
> @@ -1967,16 +1967,26 @@ static int ksz9477_config_init(struct
> phy_device *phydev)
>                         return err;
>         }
>
> +       return err;
> +}
> +
> +static int ksz9477_config_init(struct phy_device *phydev)
> +{
> +       int err;
> +
> +       /* Only KSZ9897 family of switches needs this fix. */
> +       if ((phydev->phy_id & 0xf) == 1) {
> +               err = ksz9477_phy_errata(phydev);
> +               if (err)
> +                       return err;
> +       }
> +
>         /* According to KSZ9477 Errata DS80000754C (Module 4) all EEE
> modes
>          * in this switch shall be regarded as broken.
>          */
>         if (phydev->dev_flags & MICREL_NO_EEE)
>                 phydev->eee_broken_modes = -1;
>
> -       err = genphy_restart_aneg(phydev);
> -       if (err)
> -               return err;
> -

ksz9477_phy_errata is setting the PHY for 100 Mbps speed with auto-
negotiation off, as the errata sheet requests before applying the
errata writes, so it seems wrong to remove this auto-negotiation
restart here as it would presumably be left in that state otherwise.
Maybe it should be moved into that function instead?

>         return kszphy_config_init(phydev);
>  }
>
> @@ -2085,6 +2095,42 @@ static int kszphy_resume(struct phy_device
> *phydev)
>         return 0;
>  }
>
> +static int ksz9477_resume(struct phy_device *phydev)
> +{
> +       int ret;
> +
> +       /* No need to initialize registers if not powered down. */
> +       ret = phy_read(phydev, MII_BMCR);
> +       if (ret < 0)
> +               return ret;
> +       if (!(ret & BMCR_PDOWN))
> +               return 0;
> +
> +       genphy_resume(phydev);
> +
> +       /* After switching from power-down to normal mode, an
> internal global
> +        * reset is automatically generated. Wait a minimum of 1 ms
> before
> +        * read/write access to the PHY registers.
> +        */
> +       usleep_range(1000, 2000);
> +
> +       /* Only KSZ9897 family of switches needs this fix. */
> +       if ((phydev->phy_id & 0xf) == 1) {
> +               ret = ksz9477_phy_errata(phydev);
> +               if (ret)
> +                       return ret;
> +       }
> +
> +       /* Enable PHY Interrupts */
> +       if (phy_interrupt_is_valid(phydev)) {
> +               phydev->interrupts = PHY_INTERRUPT_ENABLED;
> +               if (phydev->drv->config_intr)
> +                       phydev->drv->config_intr(phydev);
> +       }
> +
> +       return 0;
> +}
> +
>  static int kszphy_probe(struct phy_device *phydev)
>  {
>         const struct kszphy_type *type = phydev->drv->driver_data;
> @@ -5493,7 +5539,7 @@ static struct phy_driver ksphy_driver[] = {
>         .config_intr    = kszphy_config_intr,
>         .handle_interrupt = kszphy_handle_interrupt,
>         .suspend        = genphy_suspend,
> -       .resume         = genphy_resume,
> +       .resume         = ksz9477_resume,
>         .get_features   = ksz9477_get_features,
>  } };
>
> --
> 2.34.1
>