|
@@ -847,9 +847,10 @@ static struct ethtool_ops bcmgenet_ethtool_ops = {
|
|
|
};
|
|
|
|
|
|
/* Power down the unimac, based on mode. */
|
|
|
-static void bcmgenet_power_down(struct bcmgenet_priv *priv,
|
|
|
+static int bcmgenet_power_down(struct bcmgenet_priv *priv,
|
|
|
enum bcmgenet_power_mode mode)
|
|
|
{
|
|
|
+ int ret = 0;
|
|
|
u32 reg;
|
|
|
|
|
|
switch (mode) {
|
|
@@ -858,7 +859,7 @@ static void bcmgenet_power_down(struct bcmgenet_priv *priv,
|
|
|
break;
|
|
|
|
|
|
case GENET_POWER_WOL_MAGIC:
|
|
|
- bcmgenet_wol_power_down_cfg(priv, mode);
|
|
|
+ ret = bcmgenet_wol_power_down_cfg(priv, mode);
|
|
|
break;
|
|
|
|
|
|
case GENET_POWER_PASSIVE:
|
|
@@ -868,11 +869,15 @@ static void bcmgenet_power_down(struct bcmgenet_priv *priv,
|
|
|
reg |= (EXT_PWR_DOWN_PHY |
|
|
|
EXT_PWR_DOWN_DLL | EXT_PWR_DOWN_BIAS);
|
|
|
bcmgenet_ext_writel(priv, reg, EXT_EXT_PWR_MGMT);
|
|
|
+
|
|
|
+ bcmgenet_phy_power_set(priv->dev, false);
|
|
|
}
|
|
|
break;
|
|
|
default:
|
|
|
break;
|
|
|
}
|
|
|
+
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
static void bcmgenet_power_up(struct bcmgenet_priv *priv,
|
|
@@ -2462,9 +2467,6 @@ static void bcmgenet_netif_start(struct net_device *dev)
|
|
|
|
|
|
umac_enable_set(priv, CMD_TX_EN | CMD_RX_EN, true);
|
|
|
|
|
|
- if (phy_is_internal(priv->phydev))
|
|
|
- bcmgenet_power_up(priv, GENET_POWER_PASSIVE);
|
|
|
-
|
|
|
netif_tx_start_all_queues(dev);
|
|
|
|
|
|
phy_start(priv->phydev);
|
|
@@ -2483,6 +2485,12 @@ static int bcmgenet_open(struct net_device *dev)
|
|
|
if (!IS_ERR(priv->clk))
|
|
|
clk_prepare_enable(priv->clk);
|
|
|
|
|
|
+ /* If this is an internal GPHY, power it back on now, before UniMAC is
|
|
|
+ * brought out of reset as absolutely no UniMAC activity is allowed
|
|
|
+ */
|
|
|
+ if (phy_is_internal(priv->phydev))
|
|
|
+ bcmgenet_power_up(priv, GENET_POWER_PASSIVE);
|
|
|
+
|
|
|
/* take MAC out of reset */
|
|
|
bcmgenet_umac_reset(priv);
|
|
|
|
|
@@ -2606,12 +2614,12 @@ static int bcmgenet_close(struct net_device *dev)
|
|
|
free_irq(priv->irq1, priv);
|
|
|
|
|
|
if (phy_is_internal(priv->phydev))
|
|
|
- bcmgenet_power_down(priv, GENET_POWER_PASSIVE);
|
|
|
+ ret = bcmgenet_power_down(priv, GENET_POWER_PASSIVE);
|
|
|
|
|
|
if (!IS_ERR(priv->clk))
|
|
|
clk_disable_unprepare(priv->clk);
|
|
|
|
|
|
- return 0;
|
|
|
+ return ret;
|
|
|
}
|
|
|
|
|
|
static void bcmgenet_timeout(struct net_device *dev)
|
|
@@ -3097,14 +3105,16 @@ static int bcmgenet_suspend(struct device *d)
|
|
|
|
|
|
/* Prepare the device for Wake-on-LAN and switch to the slow clock */
|
|
|
if (device_may_wakeup(d) && priv->wolopts) {
|
|
|
- bcmgenet_power_down(priv, GENET_POWER_WOL_MAGIC);
|
|
|
+ ret = bcmgenet_power_down(priv, GENET_POWER_WOL_MAGIC);
|
|
|
clk_prepare_enable(priv->clk_wol);
|
|
|
+ } else if (phy_is_internal(priv->phydev)) {
|
|
|
+ ret = bcmgenet_power_down(priv, GENET_POWER_PASSIVE);
|
|
|
}
|
|
|
|
|
|
/* Turn off the clocks */
|
|
|
clk_disable_unprepare(priv->clk);
|
|
|
|
|
|
- return 0;
|
|
|
+ return ret;
|
|
|
}
|
|
|
|
|
|
static int bcmgenet_resume(struct device *d)
|
|
@@ -3123,6 +3133,12 @@ static int bcmgenet_resume(struct device *d)
|
|
|
if (ret)
|
|
|
return ret;
|
|
|
|
|
|
+ /* If this is an internal GPHY, power it back on now, before UniMAC is
|
|
|
+ * brought out of reset as absolutely no UniMAC activity is allowed
|
|
|
+ */
|
|
|
+ if (phy_is_internal(priv->phydev))
|
|
|
+ bcmgenet_power_up(priv, GENET_POWER_PASSIVE);
|
|
|
+
|
|
|
bcmgenet_umac_reset(priv);
|
|
|
|
|
|
ret = init_umac(priv);
|