Commit 578c6d28 authored by Florian Fainelli's avatar Florian Fainelli Committed by Greg Kroah-Hartman

net: systemport: Fix WoL with password after deep sleep

[ Upstream commit 8dfb8d2c ]

Broadcom STB chips support a deep sleep mode where all register
contents are lost. Because we were stashing the MagicPacket password
into some of these registers a suspend into that deep sleep then a
resumption would not lead to being able to wake-up from MagicPacket with
password again.

Fix this by keeping a software copy of the password and program it
during suspend.

Fixes: 83e82f4c ("net: systemport: add Wake-on-LAN support")
Signed-off-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent b28b106b
...@@ -519,7 +519,6 @@ static void bcm_sysport_get_wol(struct net_device *dev, ...@@ -519,7 +519,6 @@ static void bcm_sysport_get_wol(struct net_device *dev,
struct ethtool_wolinfo *wol) struct ethtool_wolinfo *wol)
{ {
struct bcm_sysport_priv *priv = netdev_priv(dev); struct bcm_sysport_priv *priv = netdev_priv(dev);
u32 reg;
wol->supported = WAKE_MAGIC | WAKE_MAGICSECURE; wol->supported = WAKE_MAGIC | WAKE_MAGICSECURE;
wol->wolopts = priv->wolopts; wol->wolopts = priv->wolopts;
...@@ -527,11 +526,7 @@ static void bcm_sysport_get_wol(struct net_device *dev, ...@@ -527,11 +526,7 @@ static void bcm_sysport_get_wol(struct net_device *dev,
if (!(priv->wolopts & WAKE_MAGICSECURE)) if (!(priv->wolopts & WAKE_MAGICSECURE))
return; return;
/* Return the programmed SecureOn password */ memcpy(wol->sopass, priv->sopass, sizeof(priv->sopass));
reg = umac_readl(priv, UMAC_PSW_MS);
put_unaligned_be16(reg, &wol->sopass[0]);
reg = umac_readl(priv, UMAC_PSW_LS);
put_unaligned_be32(reg, &wol->sopass[2]);
} }
static int bcm_sysport_set_wol(struct net_device *dev, static int bcm_sysport_set_wol(struct net_device *dev,
...@@ -547,13 +542,8 @@ static int bcm_sysport_set_wol(struct net_device *dev, ...@@ -547,13 +542,8 @@ static int bcm_sysport_set_wol(struct net_device *dev,
if (wol->wolopts & ~supported) if (wol->wolopts & ~supported)
return -EINVAL; return -EINVAL;
/* Program the SecureOn password */ if (wol->wolopts & WAKE_MAGICSECURE)
if (wol->wolopts & WAKE_MAGICSECURE) { memcpy(priv->sopass, wol->sopass, sizeof(priv->sopass));
umac_writel(priv, get_unaligned_be16(&wol->sopass[0]),
UMAC_PSW_MS);
umac_writel(priv, get_unaligned_be32(&wol->sopass[2]),
UMAC_PSW_LS);
}
/* Flag the device and relevant IRQ as wakeup capable */ /* Flag the device and relevant IRQ as wakeup capable */
if (wol->wolopts) { if (wol->wolopts) {
...@@ -2221,12 +2211,17 @@ static int bcm_sysport_suspend_to_wol(struct bcm_sysport_priv *priv) ...@@ -2221,12 +2211,17 @@ static int bcm_sysport_suspend_to_wol(struct bcm_sysport_priv *priv)
unsigned int timeout = 1000; unsigned int timeout = 1000;
u32 reg; u32 reg;
/* Password has already been programmed */
reg = umac_readl(priv, UMAC_MPD_CTRL); reg = umac_readl(priv, UMAC_MPD_CTRL);
reg |= MPD_EN; reg |= MPD_EN;
reg &= ~PSW_EN; reg &= ~PSW_EN;
if (priv->wolopts & WAKE_MAGICSECURE) if (priv->wolopts & WAKE_MAGICSECURE) {
/* Program the SecureOn password */
umac_writel(priv, get_unaligned_be16(&priv->sopass[0]),
UMAC_PSW_MS);
umac_writel(priv, get_unaligned_be32(&priv->sopass[2]),
UMAC_PSW_LS);
reg |= PSW_EN; reg |= PSW_EN;
}
umac_writel(priv, reg, UMAC_MPD_CTRL); umac_writel(priv, reg, UMAC_MPD_CTRL);
/* Make sure RBUF entered WoL mode as result */ /* Make sure RBUF entered WoL mode as result */
......
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
#ifndef __BCM_SYSPORT_H #ifndef __BCM_SYSPORT_H
#define __BCM_SYSPORT_H #define __BCM_SYSPORT_H
#include <linux/ethtool.h>
#include <linux/if_vlan.h> #include <linux/if_vlan.h>
/* Receive/transmit descriptor format */ /* Receive/transmit descriptor format */
...@@ -754,6 +755,7 @@ struct bcm_sysport_priv { ...@@ -754,6 +755,7 @@ struct bcm_sysport_priv {
unsigned int crc_fwd:1; unsigned int crc_fwd:1;
u16 rev; u16 rev;
u32 wolopts; u32 wolopts;
u8 sopass[SOPASS_MAX];
unsigned int wol_irq_disabled:1; unsigned int wol_irq_disabled:1;
/* MIB related fields */ /* MIB related fields */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment