Commit 93c52dd0 authored by Alexander Duyck's avatar Alexander Duyck Committed by Jeff Kirsher

ixgbe: Merge watchdog functionality into service task

This patch is meant to merge the functionality of the ixgbe watchdog task
into the service task.  By doing this all link state functionality will be
controlled by a single task.  As a result the reliability of the interface
will be improved as the likelihood of any race conditions is further
reduced.
Signed-off-by: default avatarAlexander Duyck <alexander.h.duyck@intel.com>
Signed-off-by: default avatarJeff Kirsher <jeffrey.t.kirsher@intel.com>
parent 7086400d
...@@ -455,11 +455,9 @@ struct ixgbe_adapter { ...@@ -455,11 +455,9 @@ struct ixgbe_adapter {
unsigned long link_check_timeout; unsigned long link_check_timeout;
struct work_struct reset_task; struct work_struct reset_task;
struct work_struct watchdog_task;
struct work_struct fdir_reinit_task; struct work_struct fdir_reinit_task;
struct work_struct check_overtemp_task; struct work_struct check_overtemp_task;
struct work_struct service_task; struct work_struct service_task;
struct timer_list watchdog_timer;
struct timer_list service_timer; struct timer_list service_timer;
u32 fdir_pballoc; u32 fdir_pballoc;
u32 atr_sample_rate; u32 atr_sample_rate;
......
...@@ -1900,7 +1900,7 @@ static void ixgbe_check_lsc(struct ixgbe_adapter *adapter) ...@@ -1900,7 +1900,7 @@ static void ixgbe_check_lsc(struct ixgbe_adapter *adapter)
if (!test_bit(__IXGBE_DOWN, &adapter->state)) { if (!test_bit(__IXGBE_DOWN, &adapter->state)) {
IXGBE_WRITE_REG(hw, IXGBE_EIMC, IXGBE_EIMC_LSC); IXGBE_WRITE_REG(hw, IXGBE_EIMC, IXGBE_EIMC_LSC);
IXGBE_WRITE_FLUSH(hw); IXGBE_WRITE_FLUSH(hw);
schedule_work(&adapter->watchdog_task); ixgbe_service_event_schedule(adapter);
} }
} }
...@@ -3940,7 +3940,6 @@ static int ixgbe_up_complete(struct ixgbe_adapter *adapter) ...@@ -3940,7 +3940,6 @@ static int ixgbe_up_complete(struct ixgbe_adapter *adapter)
* link up interrupt but shouldn't be a problem */ * link up interrupt but shouldn't be a problem */
adapter->flags |= IXGBE_FLAG_NEED_LINK_UPDATE; adapter->flags |= IXGBE_FLAG_NEED_LINK_UPDATE;
adapter->link_check_timeout = jiffies; adapter->link_check_timeout = jiffies;
mod_timer(&adapter->watchdog_timer, jiffies);
mod_timer(&adapter->service_timer, jiffies); mod_timer(&adapter->service_timer, jiffies);
/* Set PF Reset Done bit so PF/VF Mail Ops can work */ /* Set PF Reset Done bit so PF/VF Mail Ops can work */
...@@ -4179,8 +4178,6 @@ void ixgbe_down(struct ixgbe_adapter *adapter) ...@@ -4179,8 +4178,6 @@ void ixgbe_down(struct ixgbe_adapter *adapter)
netif_tx_stop_all_queues(netdev); netif_tx_stop_all_queues(netdev);
del_timer_sync(&adapter->watchdog_timer);
cancel_work_sync(&adapter->watchdog_task);
/* call carrier off first to avoid false dev_watchdog timeouts */ /* call carrier off first to avoid false dev_watchdog timeouts */
netif_carrier_off(netdev); netif_carrier_off(netdev);
netif_tx_disable(netdev); netif_tx_disable(netdev);
...@@ -5956,54 +5953,6 @@ void ixgbe_update_stats(struct ixgbe_adapter *adapter) ...@@ -5956,54 +5953,6 @@ void ixgbe_update_stats(struct ixgbe_adapter *adapter)
netdev->stats.rx_missed_errors = total_mpc; netdev->stats.rx_missed_errors = total_mpc;
} }
/**
* ixgbe_watchdog - Timer Call-back
* @data: pointer to adapter cast into an unsigned long
**/
static void ixgbe_watchdog(unsigned long data)
{
struct ixgbe_adapter *adapter = (struct ixgbe_adapter *)data;
struct ixgbe_hw *hw = &adapter->hw;
u64 eics = 0;
int i;
/*
* Do the watchdog outside of interrupt context due to the lovely
* delays that some of the newer hardware requires
*/
if (test_bit(__IXGBE_DOWN, &adapter->state))
goto watchdog_short_circuit;
if (!(adapter->flags & IXGBE_FLAG_MSIX_ENABLED)) {
/*
* for legacy and MSI interrupts don't set any bits
* that are enabled for EIAM, because this operation
* would set *both* EIMS and EICS for any bit in EIAM
*/
IXGBE_WRITE_REG(hw, IXGBE_EICS,
(IXGBE_EICS_TCP_TIMER | IXGBE_EICS_OTHER));
goto watchdog_reschedule;
}
/* get one bit for every active tx/rx interrupt vector */
for (i = 0; i < adapter->num_msix_vectors - NON_Q_VECTORS; i++) {
struct ixgbe_q_vector *qv = adapter->q_vector[i];
if (qv->rxr_count || qv->txr_count)
eics |= ((u64)1 << i);
}
/* Cause software interrupt to ensure rx rings are cleaned */
ixgbe_irq_rearm_queues(adapter, eics);
watchdog_reschedule:
/* Reset the timer */
mod_timer(&adapter->watchdog_timer, round_jiffies(jiffies + 2 * HZ));
watchdog_short_circuit:
schedule_work(&adapter->watchdog_task);
}
/** /**
* ixgbe_fdir_reinit_task - worker thread to reinit FDIR filter table * ixgbe_fdir_reinit_task - worker thread to reinit FDIR filter table
* @work: pointer to work_struct containing our data * @work: pointer to work_struct containing our data
...@@ -6028,63 +5977,83 @@ static void ixgbe_fdir_reinit_task(struct work_struct *work) ...@@ -6028,63 +5977,83 @@ static void ixgbe_fdir_reinit_task(struct work_struct *work)
netif_tx_start_all_queues(adapter->netdev); netif_tx_start_all_queues(adapter->netdev);
} }
static void ixgbe_spoof_check(struct ixgbe_adapter *adapter) /**
* ixgbe_check_hang_subtask - check for hung queues and dropped interrupts
* @adapter - pointer to the device adapter structure
*
* This function serves two purposes. First it strobes the interrupt lines
* in order to make certain interrupts are occuring. Secondly it sets the
* bits needed to check for TX hangs. As a result we should immediately
* determine if a hang has occured.
*/
static void ixgbe_check_hang_subtask(struct ixgbe_adapter *adapter)
{ {
u32 ssvpc; struct ixgbe_hw *hw = &adapter->hw;
u64 eics = 0;
int i;
/* Do not perform spoof check for 82598 */ /* If we're down or resetting, just bail */
if (adapter->hw.mac.type == ixgbe_mac_82598EB) if (test_bit(__IXGBE_DOWN, &adapter->state) ||
test_bit(__IXGBE_RESETTING, &adapter->state))
return; return;
ssvpc = IXGBE_READ_REG(&adapter->hw, IXGBE_SSVPC); /* Force detection of hung controller */
if (netif_carrier_ok(adapter->netdev)) {
for (i = 0; i < adapter->num_tx_queues; i++)
set_check_for_tx_hang(adapter->tx_ring[i]);
}
if (!(adapter->flags & IXGBE_FLAG_MSIX_ENABLED)) {
/* /*
* ssvpc register is cleared on read, if zero then no * for legacy and MSI interrupts don't set any bits
* spoofed packets in the last interval. * that are enabled for EIAM, because this operation
* would set *both* EIMS and EICS for any bit in EIAM
*/ */
if (!ssvpc) IXGBE_WRITE_REG(hw, IXGBE_EICS,
return; (IXGBE_EICS_TCP_TIMER | IXGBE_EICS_OTHER));
} else {
/* get one bit for every active tx/rx interrupt vector */
for (i = 0; i < adapter->num_msix_vectors - NON_Q_VECTORS; i++) {
struct ixgbe_q_vector *qv = adapter->q_vector[i];
if (qv->rxr_count || qv->txr_count)
eics |= ((u64)1 << i);
}
}
e_warn(drv, "%d Spoofed packets detected\n", ssvpc); /* Cause software interrupt to ensure rings are cleaned */
} ixgbe_irq_rearm_queues(adapter, eics);
static DEFINE_MUTEX(ixgbe_watchdog_lock); }
/** /**
* ixgbe_watchdog_task - worker thread to bring link up * ixgbe_watchdog_update_link - update the link status
* @work: pointer to work_struct containing our data * @adapter - pointer to the device adapter structure
* @link_speed - pointer to a u32 to store the link_speed
**/ **/
static void ixgbe_watchdog_task(struct work_struct *work) static void ixgbe_watchdog_update_link(struct ixgbe_adapter *adapter)
{ {
struct ixgbe_adapter *adapter = container_of(work,
struct ixgbe_adapter,
watchdog_task);
struct net_device *netdev = adapter->netdev;
struct ixgbe_hw *hw = &adapter->hw; struct ixgbe_hw *hw = &adapter->hw;
u32 link_speed; u32 link_speed = adapter->link_speed;
bool link_up; bool link_up = adapter->link_up;
int i; int i;
struct ixgbe_ring *tx_ring;
int some_tx_pending = 0;
mutex_lock(&ixgbe_watchdog_lock);
link_up = adapter->link_up; if (!(adapter->flags & IXGBE_FLAG_NEED_LINK_UPDATE))
link_speed = adapter->link_speed; return;
if (adapter->flags & IXGBE_FLAG_NEED_LINK_UPDATE) { if (hw->mac.ops.check_link) {
hw->mac.ops.check_link(hw, &link_speed, &link_up, false); hw->mac.ops.check_link(hw, &link_speed, &link_up, false);
} else {
/* always assume link is up, if no check link function */
link_speed = IXGBE_LINK_SPEED_10GB_FULL;
link_up = true;
}
if (link_up) { if (link_up) {
#ifdef CONFIG_DCB
if (adapter->flags & IXGBE_FLAG_DCB_ENABLED) { if (adapter->flags & IXGBE_FLAG_DCB_ENABLED) {
for (i = 0; i < MAX_TRAFFIC_CLASS; i++) for (i = 0; i < MAX_TRAFFIC_CLASS; i++)
hw->mac.ops.fc_enable(hw, i); hw->mac.ops.fc_enable(hw, i);
} else { } else {
hw->mac.ops.fc_enable(hw, 0); hw->mac.ops.fc_enable(hw, 0);
} }
#else
hw->mac.ops.fc_enable(hw, 0);
#endif
} }
if (link_up || if (link_up ||
...@@ -6092,15 +6061,31 @@ static void ixgbe_watchdog_task(struct work_struct *work) ...@@ -6092,15 +6061,31 @@ static void ixgbe_watchdog_task(struct work_struct *work)
IXGBE_TRY_LINK_TIMEOUT))) { IXGBE_TRY_LINK_TIMEOUT))) {
adapter->flags &= ~IXGBE_FLAG_NEED_LINK_UPDATE; adapter->flags &= ~IXGBE_FLAG_NEED_LINK_UPDATE;
IXGBE_WRITE_REG(hw, IXGBE_EIMS, IXGBE_EIMC_LSC); IXGBE_WRITE_REG(hw, IXGBE_EIMS, IXGBE_EIMC_LSC);
IXGBE_WRITE_FLUSH(hw);
} }
adapter->link_up = link_up; adapter->link_up = link_up;
adapter->link_speed = link_speed; adapter->link_speed = link_speed;
} }
if (link_up) { /**
if (!netif_carrier_ok(netdev)) { * ixgbe_watchdog_link_is_up - update netif_carrier status and
* print link up message
* @adapter - pointer to the device adapter structure
**/
static void ixgbe_watchdog_link_is_up(struct ixgbe_adapter *adapter)
{
struct net_device *netdev = adapter->netdev;
struct ixgbe_hw *hw = &adapter->hw;
u32 link_speed = adapter->link_speed;
bool flow_rx, flow_tx; bool flow_rx, flow_tx;
/* only continue if link was previously down */
if (netif_carrier_ok(netdev))
return;
adapter->flags2 &= ~IXGBE_FLAG2_SEARCH_FOR_SFP;
switch (hw->mac.type) { switch (hw->mac.type) {
case ixgbe_mac_82598EB: { case ixgbe_mac_82598EB: {
u32 frctl = IXGBE_READ_REG(hw, IXGBE_FCTRL); u32 frctl = IXGBE_READ_REG(hw, IXGBE_FCTRL);
...@@ -6109,8 +6094,8 @@ static void ixgbe_watchdog_task(struct work_struct *work) ...@@ -6109,8 +6094,8 @@ static void ixgbe_watchdog_task(struct work_struct *work)
flow_tx = !!(rmcs & IXGBE_RMCS_TFCE_802_3X); flow_tx = !!(rmcs & IXGBE_RMCS_TFCE_802_3X);
} }
break; break;
case ixgbe_mac_82599EB: case ixgbe_mac_X540:
case ixgbe_mac_X540: { case ixgbe_mac_82599EB: {
u32 mflcn = IXGBE_READ_REG(hw, IXGBE_MFLCN); u32 mflcn = IXGBE_READ_REG(hw, IXGBE_MFLCN);
u32 fccfg = IXGBE_READ_REG(hw, IXGBE_FCCFG); u32 fccfg = IXGBE_READ_REG(hw, IXGBE_FCCFG);
flow_rx = !!(mflcn & IXGBE_MFLCN_RFCE); flow_rx = !!(mflcn & IXGBE_MFLCN_RFCE);
...@@ -6122,7 +6107,6 @@ static void ixgbe_watchdog_task(struct work_struct *work) ...@@ -6122,7 +6107,6 @@ static void ixgbe_watchdog_task(struct work_struct *work)
flow_rx = false; flow_rx = false;
break; break;
} }
e_info(drv, "NIC Link is Up %s, Flow Control: %s\n", e_info(drv, "NIC Link is Up %s, Flow Control: %s\n",
(link_speed == IXGBE_LINK_SPEED_10GB_FULL ? (link_speed == IXGBE_LINK_SPEED_10GB_FULL ?
"10 Gbps" : "10 Gbps" :
...@@ -6136,26 +6120,48 @@ static void ixgbe_watchdog_task(struct work_struct *work) ...@@ -6136,26 +6120,48 @@ static void ixgbe_watchdog_task(struct work_struct *work)
(flow_tx ? "TX" : "None")))); (flow_tx ? "TX" : "None"))));
netif_carrier_on(netdev); netif_carrier_on(netdev);
#ifdef HAVE_IPLINK_VF_CONFIG
ixgbe_check_vf_rate_limit(adapter); ixgbe_check_vf_rate_limit(adapter);
} else { #endif /* HAVE_IPLINK_VF_CONFIG */
/* Force detection of hung controller */ }
for (i = 0; i < adapter->num_tx_queues; i++) {
tx_ring = adapter->tx_ring[i]; /**
set_check_for_tx_hang(tx_ring); * ixgbe_watchdog_link_is_down - update netif_carrier status and
} * print link down message
} * @adapter - pointer to the adapter structure
} else { **/
static void ixgbe_watchdog_link_is_down(struct ixgbe_adapter* adapter)
{
struct net_device *netdev = adapter->netdev;
struct ixgbe_hw *hw = &adapter->hw;
adapter->link_up = false; adapter->link_up = false;
adapter->link_speed = 0; adapter->link_speed = 0;
if (netif_carrier_ok(netdev)) {
/* only continue if link was up previously */
if (!netif_carrier_ok(netdev))
return;
/* poll for SFP+ cable when link is down */
if (ixgbe_is_sfp(hw) && hw->mac.type == ixgbe_mac_82598EB)
adapter->flags2 |= IXGBE_FLAG2_SEARCH_FOR_SFP;
e_info(drv, "NIC Link is Down\n"); e_info(drv, "NIC Link is Down\n");
netif_carrier_off(netdev); netif_carrier_off(netdev);
} }
}
if (!netif_carrier_ok(netdev)) { /**
* ixgbe_watchdog_flush_tx - flush queues on link down
* @adapter - pointer to the device adapter structure
**/
static void ixgbe_watchdog_flush_tx(struct ixgbe_adapter *adapter)
{
int i;
int some_tx_pending = 0;
if (!netif_carrier_ok(adapter->netdev)) {
for (i = 0; i < adapter->num_tx_queues; i++) { for (i = 0; i < adapter->num_tx_queues; i++) {
tx_ring = adapter->tx_ring[i]; struct ixgbe_ring *tx_ring = adapter->tx_ring[i];
if (tx_ring->next_to_use != tx_ring->next_to_clean) { if (tx_ring->next_to_use != tx_ring->next_to_clean) {
some_tx_pending = 1; some_tx_pending = 1;
break; break;
...@@ -6171,10 +6177,49 @@ static void ixgbe_watchdog_task(struct work_struct *work) ...@@ -6171,10 +6177,49 @@ static void ixgbe_watchdog_task(struct work_struct *work)
schedule_work(&adapter->reset_task); schedule_work(&adapter->reset_task);
} }
} }
}
static void ixgbe_spoof_check(struct ixgbe_adapter *adapter)
{
u32 ssvpc;
/* Do not perform spoof check for 82598 */
if (adapter->hw.mac.type == ixgbe_mac_82598EB)
return;
ssvpc = IXGBE_READ_REG(&adapter->hw, IXGBE_SSVPC);
/*
* ssvpc register is cleared on read, if zero then no
* spoofed packets in the last interval.
*/
if (!ssvpc)
return;
e_warn(drv, "%d Spoofed packets detected\n", ssvpc);
}
/**
* ixgbe_watchdog_subtask - check and bring link up
* @adapter - pointer to the device adapter structure
**/
static void ixgbe_watchdog_subtask(struct ixgbe_adapter *adapter)
{
/* if interface is down do nothing */
if (test_bit(__IXGBE_DOWN, &adapter->state))
return;
ixgbe_watchdog_update_link(adapter);
if (adapter->link_up)
ixgbe_watchdog_link_is_up(adapter);
else
ixgbe_watchdog_link_is_down(adapter);
ixgbe_spoof_check(adapter); ixgbe_spoof_check(adapter);
ixgbe_update_stats(adapter); ixgbe_update_stats(adapter);
mutex_unlock(&ixgbe_watchdog_lock);
ixgbe_watchdog_flush_tx(adapter);
} }
/** /**
...@@ -6308,6 +6353,8 @@ static void ixgbe_service_task(struct work_struct *work) ...@@ -6308,6 +6353,8 @@ static void ixgbe_service_task(struct work_struct *work)
ixgbe_sfp_detection_subtask(adapter); ixgbe_sfp_detection_subtask(adapter);
ixgbe_sfp_link_config_subtask(adapter); ixgbe_sfp_link_config_subtask(adapter);
ixgbe_watchdog_subtask(adapter);
ixgbe_check_hang_subtask(adapter);
ixgbe_service_event_complete(adapter); ixgbe_service_event_complete(adapter);
} }
...@@ -7485,12 +7532,8 @@ static int __devinit ixgbe_probe(struct pci_dev *pdev, ...@@ -7485,12 +7532,8 @@ static int __devinit ixgbe_probe(struct pci_dev *pdev,
setup_timer(&adapter->service_timer, &ixgbe_service_timer, setup_timer(&adapter->service_timer, &ixgbe_service_timer,
(unsigned long) adapter); (unsigned long) adapter);
init_timer(&adapter->watchdog_timer);
adapter->watchdog_timer.function = ixgbe_watchdog;
adapter->watchdog_timer.data = (unsigned long)adapter;
INIT_WORK(&adapter->reset_task, ixgbe_reset_task); INIT_WORK(&adapter->reset_task, ixgbe_reset_task);
INIT_WORK(&adapter->watchdog_task, ixgbe_watchdog_task);
INIT_WORK(&adapter->service_task, ixgbe_service_task); INIT_WORK(&adapter->service_task, ixgbe_service_task);
clear_bit(__IXGBE_SERVICE_SCHED, &adapter->state); clear_bit(__IXGBE_SERVICE_SCHED, &adapter->state);
...@@ -7643,13 +7686,6 @@ static void __devexit ixgbe_remove(struct pci_dev *pdev) ...@@ -7643,13 +7686,6 @@ static void __devexit ixgbe_remove(struct pci_dev *pdev)
set_bit(__IXGBE_DOWN, &adapter->state); set_bit(__IXGBE_DOWN, &adapter->state);
cancel_work_sync(&adapter->service_task); cancel_work_sync(&adapter->service_task);
/*
* The timers may be rescheduled, so explicitly disable them
* from being rescheduled.
*/
del_timer_sync(&adapter->watchdog_timer);
cancel_work_sync(&adapter->watchdog_task);
if (adapter->flags & IXGBE_FLAG_FDIR_HASH_CAPABLE || if (adapter->flags & IXGBE_FLAG_FDIR_HASH_CAPABLE ||
adapter->flags & IXGBE_FLAG_FDIR_PERFECT_CAPABLE) adapter->flags & IXGBE_FLAG_FDIR_PERFECT_CAPABLE)
cancel_work_sync(&adapter->fdir_reinit_task); cancel_work_sync(&adapter->fdir_reinit_task);
......
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