dl2k: Implement suspend
Add suspend/resume support to dl2k driver. This requires RX/TX rings to be reset so split out the required functionality from alloc_list() into new rio_reset_ring(). Tested on Asus NX1101 (IP1000A) and D-Link DGE-550T (DL-2000). Signed-off-by: Ondrej Zary <linux@rainbow-software.org> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
966e07f4bf
commit
1777ddb84a
@ -464,36 +464,40 @@ static void free_list(struct net_device *dev)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void rio_reset_ring(struct netdev_private *np)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
np->cur_rx = 0;
|
||||||
|
np->cur_tx = 0;
|
||||||
|
np->old_rx = 0;
|
||||||
|
np->old_tx = 0;
|
||||||
|
|
||||||
|
for (i = 0; i < TX_RING_SIZE; i++)
|
||||||
|
np->tx_ring[i].status = cpu_to_le64(TFDDone);
|
||||||
|
|
||||||
|
for (i = 0; i < RX_RING_SIZE; i++)
|
||||||
|
np->rx_ring[i].status = 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* allocate and initialize Tx and Rx descriptors */
|
/* allocate and initialize Tx and Rx descriptors */
|
||||||
static int alloc_list(struct net_device *dev)
|
static int alloc_list(struct net_device *dev)
|
||||||
{
|
{
|
||||||
struct netdev_private *np = netdev_priv(dev);
|
struct netdev_private *np = netdev_priv(dev);
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
np->cur_rx = np->cur_tx = 0;
|
rio_reset_ring(np);
|
||||||
np->old_rx = np->old_tx = 0;
|
|
||||||
np->rx_buf_sz = (dev->mtu <= 1500 ? PACKET_SIZE : dev->mtu + 32);
|
np->rx_buf_sz = (dev->mtu <= 1500 ? PACKET_SIZE : dev->mtu + 32);
|
||||||
|
|
||||||
/* Initialize Tx descriptors, TFDListPtr leaves in start_xmit(). */
|
/* Initialize Tx descriptors, TFDListPtr leaves in start_xmit(). */
|
||||||
for (i = 0; i < TX_RING_SIZE; i++) {
|
for (i = 0; i < TX_RING_SIZE; i++) {
|
||||||
np->tx_skbuff[i] = NULL;
|
np->tx_skbuff[i] = NULL;
|
||||||
np->tx_ring[i].status = cpu_to_le64(TFDDone);
|
|
||||||
np->tx_ring[i].next_desc = cpu_to_le64(np->tx_ring_dma +
|
np->tx_ring[i].next_desc = cpu_to_le64(np->tx_ring_dma +
|
||||||
((i + 1) % TX_RING_SIZE) *
|
((i + 1) % TX_RING_SIZE) *
|
||||||
sizeof(struct netdev_desc));
|
sizeof(struct netdev_desc));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Initialize Rx descriptors */
|
/* Initialize Rx descriptors & allocate buffers */
|
||||||
for (i = 0; i < RX_RING_SIZE; i++) {
|
|
||||||
np->rx_ring[i].next_desc = cpu_to_le64(np->rx_ring_dma +
|
|
||||||
((i + 1) % RX_RING_SIZE) *
|
|
||||||
sizeof(struct netdev_desc));
|
|
||||||
np->rx_ring[i].status = 0;
|
|
||||||
np->rx_ring[i].fraginfo = 0;
|
|
||||||
np->rx_skbuff[i] = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Allocate the rx buffers */
|
|
||||||
for (i = 0; i < RX_RING_SIZE; i++) {
|
for (i = 0; i < RX_RING_SIZE; i++) {
|
||||||
/* Allocated fixed size of skbuff */
|
/* Allocated fixed size of skbuff */
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
@ -505,6 +509,9 @@ static int alloc_list(struct net_device *dev)
|
|||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
np->rx_ring[i].next_desc = cpu_to_le64(np->rx_ring_dma +
|
||||||
|
((i + 1) % RX_RING_SIZE) *
|
||||||
|
sizeof(struct netdev_desc));
|
||||||
/* Rubicon now supports 40 bits of addressing space. */
|
/* Rubicon now supports 40 bits of addressing space. */
|
||||||
np->rx_ring[i].fraginfo =
|
np->rx_ring[i].fraginfo =
|
||||||
cpu_to_le64(pci_map_single(
|
cpu_to_le64(pci_map_single(
|
||||||
@ -1824,11 +1831,55 @@ rio_remove1 (struct pci_dev *pdev)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_PM_SLEEP
|
||||||
|
static int rio_suspend(struct device *device)
|
||||||
|
{
|
||||||
|
struct net_device *dev = dev_get_drvdata(device);
|
||||||
|
struct netdev_private *np = netdev_priv(dev);
|
||||||
|
|
||||||
|
if (!netif_running(dev))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
netif_device_detach(dev);
|
||||||
|
del_timer_sync(&np->timer);
|
||||||
|
rio_hw_stop(dev);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int rio_resume(struct device *device)
|
||||||
|
{
|
||||||
|
struct net_device *dev = dev_get_drvdata(device);
|
||||||
|
struct netdev_private *np = netdev_priv(dev);
|
||||||
|
|
||||||
|
if (!netif_running(dev))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
rio_reset_ring(np);
|
||||||
|
rio_hw_init(dev);
|
||||||
|
np->timer.expires = jiffies + 1 * HZ;
|
||||||
|
add_timer(&np->timer);
|
||||||
|
netif_device_attach(dev);
|
||||||
|
dl2k_enable_int(np);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static SIMPLE_DEV_PM_OPS(rio_pm_ops, rio_suspend, rio_resume);
|
||||||
|
#define RIO_PM_OPS (&rio_pm_ops)
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
#define RIO_PM_OPS NULL
|
||||||
|
|
||||||
|
#endif /* CONFIG_PM_SLEEP */
|
||||||
|
|
||||||
static struct pci_driver rio_driver = {
|
static struct pci_driver rio_driver = {
|
||||||
.name = "dl2k",
|
.name = "dl2k",
|
||||||
.id_table = rio_pci_tbl,
|
.id_table = rio_pci_tbl,
|
||||||
.probe = rio_probe1,
|
.probe = rio_probe1,
|
||||||
.remove = rio_remove1,
|
.remove = rio_remove1,
|
||||||
|
.driver.pm = RIO_PM_OPS,
|
||||||
};
|
};
|
||||||
|
|
||||||
module_pci_driver(rio_driver);
|
module_pci_driver(rio_driver);
|
||||||
|
Loading…
Reference in New Issue
Block a user