OSDN Git Service

net: phy: broadcom: Implement suspend/resume for AC131 and BCM5241
authorFlorian Fainelli <f.fainelli@gmail.com>
Mon, 15 Aug 2022 19:07:47 +0000 (12:07 -0700)
committerDavid S. Miller <davem@davemloft.net>
Wed, 17 Aug 2022 10:49:23 +0000 (11:49 +0100)
Implement the suspend/resume procedure for the Broadcom AC131 and BCM5241 type
of PHYs (10/100 only) by entering the standard power down followed by the
proprietary standby mode in the auxiliary mode 4 shadow register. On resume,
the PHY software reset is enough to make it come out of standby mode so we can
utilize brcm_fet_config_init() as the resume hook.

Signed-off-by: Florian Fainelli <f.fainelli@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/phy/broadcom.c
include/linux/brcmphy.h

index 31fbcdd..ad71c88 100644 (file)
@@ -766,6 +766,41 @@ static irqreturn_t brcm_fet_handle_interrupt(struct phy_device *phydev)
        return IRQ_HANDLED;
 }
 
+static int brcm_fet_suspend(struct phy_device *phydev)
+{
+       int reg, err, err2, brcmtest;
+
+       /* We cannot use a read/modify/write here otherwise the PHY continues
+        * to drive LEDs which defeats the purpose of low power mode.
+        */
+       err = phy_write(phydev, MII_BMCR, BMCR_PDOWN);
+       if (err < 0)
+               return err;
+
+       /* Enable shadow register access */
+       brcmtest = phy_read(phydev, MII_BRCM_FET_BRCMTEST);
+       if (brcmtest < 0)
+               return brcmtest;
+
+       reg = brcmtest | MII_BRCM_FET_BT_SRE;
+
+       err = phy_write(phydev, MII_BRCM_FET_BRCMTEST, reg);
+       if (err < 0)
+               return err;
+
+       /* Set standby mode */
+       err = phy_modify(phydev, MII_BRCM_FET_SHDW_AUXMODE4,
+                        MII_BRCM_FET_SHDW_AM4_STANDBY,
+                        MII_BRCM_FET_SHDW_AM4_STANDBY);
+
+       /* Disable shadow register access */
+       err2 = phy_write(phydev, MII_BRCM_FET_BRCMTEST, brcmtest);
+       if (!err)
+               err = err2;
+
+       return err;
+}
+
 static int bcm54xx_phy_probe(struct phy_device *phydev)
 {
        struct bcm54xx_phy_priv *priv;
@@ -1033,6 +1068,8 @@ static struct phy_driver broadcom_drivers[] = {
        .config_init    = brcm_fet_config_init,
        .config_intr    = brcm_fet_config_intr,
        .handle_interrupt = brcm_fet_handle_interrupt,
+       .suspend        = brcm_fet_suspend,
+       .resume         = brcm_fet_config_init,
 }, {
        .phy_id         = PHY_ID_BCM5241,
        .phy_id_mask    = 0xfffffff0,
@@ -1041,6 +1078,8 @@ static struct phy_driver broadcom_drivers[] = {
        .config_init    = brcm_fet_config_init,
        .config_intr    = brcm_fet_config_intr,
        .handle_interrupt = brcm_fet_handle_interrupt,
+       .suspend        = brcm_fet_suspend,
+       .resume         = brcm_fet_config_init,
 }, {
        .phy_id         = PHY_ID_BCM5395,
        .phy_id_mask    = 0xfffffff0,
index 6ff567e..9e77165 100644 (file)
 #define MII_BRCM_FET_SHDW_MC_FAME      0x4000  /* Force Auto MDIX enable */
 
 #define MII_BRCM_FET_SHDW_AUXMODE4     0x1a    /* Auxiliary mode 4 */
+#define MII_BRCM_FET_SHDW_AM4_STANDBY  0x0008  /* Standby enable */
 #define MII_BRCM_FET_SHDW_AM4_LED_MASK 0x0003
 #define MII_BRCM_FET_SHDW_AM4_LED_MODE1 0x0001