Commit e5ea8da0 authored by Mithlesh Thukral's avatar Mithlesh Thukral Committed by Greg Kroah-Hartman

Staging: sxg: Add watchdog timer for managing Link states for SXG driver

Add a watchdog timer to take care of link change notifications.
Link changes would now be handled asynchronously as they involve large delays.
Signed-off-by: default avatarLinSysSoft Sahara Team <saharaproj@linsyssoft.com>
Signed-off-by: default avatarMithlesh Thukral <mithlesh@linsyssoft.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 524ca9c1
...@@ -136,6 +136,9 @@ static int sxg_register_interrupt(struct adapter_t *adapter); ...@@ -136,6 +136,9 @@ static int sxg_register_interrupt(struct adapter_t *adapter);
static void sxg_remove_isr(struct adapter_t *adapter); static void sxg_remove_isr(struct adapter_t *adapter);
static irqreturn_t sxg_isr(int irq, void *dev_id); static irqreturn_t sxg_isr(int irq, void *dev_id);
static void sxg_watchdog(unsigned long data);
static void sxg_update_link_status (struct work_struct *work);
#define XXXTODO 0 #define XXXTODO 0
#if XXXTODO #if XXXTODO
...@@ -1122,6 +1125,12 @@ static int sxg_entry_probe(struct pci_dev *pcidev, ...@@ -1122,6 +1125,12 @@ static int sxg_entry_probe(struct pci_dev *pcidev,
netif_napi_add(netdev, &adapter->napi, netif_napi_add(netdev, &adapter->napi,
sxg_poll, SXG_NETDEV_WEIGHT); sxg_poll, SXG_NETDEV_WEIGHT);
netdev->watchdog_timeo = 2 * HZ;
init_timer(&adapter->watchdog_timer);
adapter->watchdog_timer.function = &sxg_watchdog;
adapter->watchdog_timer.data = (unsigned long) adapter;
INIT_WORK(&adapter->update_link_status, sxg_update_link_status);
DBG_ERROR DBG_ERROR
("sxg: %s addr 0x%lx, irq %d, MAC addr \ ("sxg: %s addr 0x%lx, irq %d, MAC addr \
%02X:%02X:%02X:%02X:%02X:%02X\n", %02X:%02X:%02X:%02X:%02X:%02X\n",
...@@ -1441,7 +1450,10 @@ static int sxg_process_isr(struct adapter_t *adapter, u32 MessageId) ...@@ -1441,7 +1450,10 @@ static int sxg_process_isr(struct adapter_t *adapter, u32 MessageId)
} }
/* Link event */ /* Link event */
if (Isr & SXG_ISR_LINK) { if (Isr & SXG_ISR_LINK) {
sxg_link_event(adapter); if (adapter->state != ADAPT_DOWN) {
adapter->link_status_changed = 1;
schedule_work(&adapter->update_link_status);
}
} }
/* Debug - breakpoint hit */ /* Debug - breakpoint hit */
if (Isr & SXG_ISR_BREAK) { if (Isr & SXG_ISR_BREAK) {
...@@ -2260,6 +2272,7 @@ int sxg_second_open(struct net_device * dev) ...@@ -2260,6 +2272,7 @@ int sxg_second_open(struct net_device * dev)
sxg_register_intr(adapter); sxg_register_intr(adapter);
spin_unlock_irqrestore(&sxg_global.driver_lock, sxg_global.flags); spin_unlock_irqrestore(&sxg_global.driver_lock, sxg_global.flags);
mod_timer(&adapter->watchdog_timer, jiffies);
return (STATUS_SUCCESS); return (STATUS_SUCCESS);
} }
...@@ -2312,27 +2325,28 @@ static int sxg_entry_halt(struct net_device *dev) ...@@ -2312,27 +2325,28 @@ static int sxg_entry_halt(struct net_device *dev)
RssIds = SXG_RSS_CPU_COUNT(adapter); RssIds = SXG_RSS_CPU_COUNT(adapter);
IsrCount = adapter->msi_enabled ? RssIds : 1; IsrCount = adapter->msi_enabled ? RssIds : 1;
/* Disable interrupts */
napi_disable(&adapter->napi);
spin_lock_irqsave(&sxg_global.driver_lock, sxg_global.flags); spin_lock_irqsave(&sxg_global.driver_lock, sxg_global.flags);
DBG_ERROR("sxg: %s (%s) ENTER\n", __func__, dev->name); SXG_DISABLE_ALL_INTERRUPTS(adapter);
WRITE_REG(adapter->UcodeRegs[0].RcvCmd, 0, true);
netif_stop_queue(adapter->netdev);
adapter->state = ADAPT_DOWN; adapter->state = ADAPT_DOWN;
adapter->linkstate = LINK_DOWN; adapter->linkstate = LINK_DOWN;
adapter->devflags_prev = 0;
DBG_ERROR("sxg: %s (%s) set adapter[%p] state to ADAPT_DOWN(%d)\n",
__func__, dev->name, adapter, adapter->state);
/* Disable interrupts */
SXG_DISABLE_ALL_INTERRUPTS(adapter);
spin_unlock_irqrestore(&sxg_global.driver_lock, sxg_global.flags); spin_unlock_irqrestore(&sxg_global.driver_lock, sxg_global.flags);
sxg_deregister_interrupt(adapter); sxg_deregister_interrupt(adapter);
WRITE_REG(HwRegs->Reset, 0xDEAD, FLUSH); WRITE_REG(HwRegs->Reset, 0xDEAD, FLUSH);
mdelay(5000); mdelay(5000);
del_timer_sync(&adapter->watchdog_timer);
netif_stop_queue(dev);
netif_carrier_off(dev);
napi_disable(&adapter->napi);
WRITE_REG(adapter->UcodeRegs[0].RcvCmd, 0, true);
adapter->devflags_prev = 0;
DBG_ERROR("sxg: %s (%s) set adapter[%p] state to ADAPT_DOWN(%d)\n",
__func__, dev->name, adapter, adapter->state);
spin_lock(&adapter->RcvQLock); spin_lock(&adapter->RcvQLock);
/* Free all the blocks and the buffers, moved from remove() routine */ /* Free all the blocks and the buffers, moved from remove() routine */
if (!(IsListEmpty(&adapter->AllRcvBlocks))) { if (!(IsListEmpty(&adapter->AllRcvBlocks))) {
...@@ -3013,6 +3027,8 @@ static void sxg_link_event(struct adapter_t *adapter) ...@@ -3013,6 +3027,8 @@ static void sxg_link_event(struct adapter_t *adapter)
int status; int status;
u32 Value; u32 Value;
if (adapter->state == ADAPT_DOWN)
return;
SXG_TRACE(TRACE_SXG, SxgTraceBuffer, TRACE_NOISY, "LinkEvnt", SXG_TRACE(TRACE_SXG, SxgTraceBuffer, TRACE_NOISY, "LinkEvnt",
adapter, 0, 0, 0); adapter, 0, 0, 0);
DBG_ERROR("ENTER %s\n", __func__); DBG_ERROR("ENTER %s\n", __func__);
...@@ -3053,10 +3069,13 @@ static void sxg_link_event(struct adapter_t *adapter) ...@@ -3053,10 +3069,13 @@ static void sxg_link_event(struct adapter_t *adapter)
sxg_link_state(adapter, LinkState); sxg_link_state(adapter, LinkState);
DBG_ERROR("SXG: Link Alarm occurred. Link is %s\n", DBG_ERROR("SXG: Link Alarm occurred. Link is %s\n",
((LinkState == SXG_LINK_UP) ? "UP" : "DOWN")); ((LinkState == SXG_LINK_UP) ? "UP" : "DOWN"));
if (LinkState == SXG_LINK_UP) if (LinkState == SXG_LINK_UP) {
netif_carrier_on(netdev); netif_carrier_on(netdev);
else netif_tx_start_all_queues(netdev);
} else {
netif_tx_stop_all_queues(netdev);
netif_carrier_off(netdev); netif_carrier_off(netdev);
}
} else { } else {
/* /*
* XXXTODO - Assuming Link Attention is only being generated * XXXTODO - Assuming Link Attention is only being generated
...@@ -4435,6 +4454,27 @@ static struct net_device_stats *sxg_get_stats(struct net_device * dev) ...@@ -4435,6 +4454,27 @@ static struct net_device_stats *sxg_get_stats(struct net_device * dev)
return (&adapter->stats); return (&adapter->stats);
} }
static void sxg_watchdog(unsigned long data)
{
struct adapter_t *adapter = (struct adapter_t *) data;
if (adapter->state != ADAPT_DOWN) {
sxg_link_event(adapter);
/* Reset the timer */
mod_timer(&adapter->watchdog_timer, round_jiffies(jiffies + 2 * HZ));
}
}
static void sxg_update_link_status (struct work_struct *work)
{
struct adapter_t *adapter = (struct adapter_t *)container_of
(work, struct adapter_t, update_link_status);
if (likely(adapter->link_status_changed)) {
sxg_link_event(adapter);
adapter->link_status_changed = 0;
}
}
static struct pci_driver sxg_driver = { static struct pci_driver sxg_driver = {
.name = sxg_driver_name, .name = sxg_driver_name,
.id_table = sxg_pci_tbl, .id_table = sxg_pci_tbl,
......
...@@ -715,6 +715,9 @@ struct adapter_t { ...@@ -715,6 +715,9 @@ struct adapter_t {
/*MSI-X related data elements*/ /*MSI-X related data elements*/
u32 nr_msix_entries; u32 nr_msix_entries;
struct msix_entry *msi_entries; struct msix_entry *msi_entries;
struct timer_list watchdog_timer;
struct work_struct update_link_status;
u32 link_status_changed;
}; };
#if SLIC_DUMP_ENABLED #if SLIC_DUMP_ENABLED
......
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