[kernel] r14769 - in dists/trunk/linux-2.6/debian: . patches/bugfix/all patches/series

Ben Hutchings benh at alioth.debian.org
Thu Dec 10 03:58:08 UTC 2009


Author: benh
Date: Thu Dec 10 03:58:05 2009
New Revision: 14769

Log:
atl1c: Fix system hang when link drops (Closes: #559577)

Added:
   dists/trunk/linux-2.6/debian/patches/bugfix/all/atl1c-use-common_task-instead-of-reset_task-and-link.patch
Modified:
   dists/trunk/linux-2.6/debian/changelog
   dists/trunk/linux-2.6/debian/patches/series/2

Modified: dists/trunk/linux-2.6/debian/changelog
==============================================================================
--- dists/trunk/linux-2.6/debian/changelog	Thu Dec 10 01:48:04 2009	(r14768)
+++ dists/trunk/linux-2.6/debian/changelog	Thu Dec 10 03:58:05 2009	(r14769)
@@ -12,6 +12,7 @@
   * aufs2: Update to snapshot from 2009-12-05
   * postinst: Fix failure paths in check for missing firmware
     (Closes: #560263)
+  * atl1c: Fix system hang when link drops (Closes: #559577)
 
  -- Bastian Blank <waldi at debian.org>  Sun, 06 Dec 2009 20:42:14 +0100
 

Added: dists/trunk/linux-2.6/debian/patches/bugfix/all/atl1c-use-common_task-instead-of-reset_task-and-link.patch
==============================================================================
--- /dev/null	00:00:00 1970	(empty, because file is newly added)
+++ dists/trunk/linux-2.6/debian/patches/bugfix/all/atl1c-use-common_task-instead-of-reset_task-and-link.patch	Thu Dec 10 03:58:05 2009	(r14769)
@@ -0,0 +1,181 @@
+From cb19054697e92a793f336380fd72c588521178ff Mon Sep 17 00:00:00 2001
+From: Jie Yang <jie.yang at atheros.com>
+Date: Sun, 6 Dec 2009 23:16:58 +0000
+Subject: [PATCH] atl1c:use common_task instead of reset_task and link_chg_task
+
+use common_task instead of reset_task and link_chg_task, so it fix "call cancel_work_sync
+from the work itself".
+
+Signed-off-by: Jie Yang <jie.yang at atheros.com>
+Signed-off-by: David S. Miller <davem at davemloft.net>
+---
+ drivers/net/atl1c/atl1c.h      |    6 ++-
+ drivers/net/atl1c/atl1c_main.c |   72 ++++++++++++++++++---------------------
+ 2 files changed, 37 insertions(+), 41 deletions(-)
+
+diff --git a/drivers/net/atl1c/atl1c.h b/drivers/net/atl1c/atl1c.h
+index 7e09084..efe5435 100644
+--- a/drivers/net/atl1c/atl1c.h
++++ b/drivers/net/atl1c/atl1c.h
+@@ -555,6 +555,9 @@ struct atl1c_adapter {
+ #define __AT_TESTING        0x0001
+ #define __AT_RESETTING      0x0002
+ #define __AT_DOWN           0x0003
++	u8 work_event;
++#define ATL1C_WORK_EVENT_RESET 		0x01
++#define ATL1C_WORK_EVENT_LINK_CHANGE	0x02
+ 	u32 msg_enable;
+ 
+ 	bool have_msi;
+@@ -566,8 +569,7 @@ struct atl1c_adapter {
+ 	spinlock_t tx_lock;
+ 	atomic_t irq_sem;
+ 
+-	struct work_struct reset_task;
+-	struct work_struct link_chg_task;
++	struct work_struct common_task;
+ 	struct timer_list watchdog_timer;
+ 	struct timer_list phy_config_timer;
+ 
+diff --git a/drivers/net/atl1c/atl1c_main.c b/drivers/net/atl1c/atl1c_main.c
+index 1098dad..666261b 100644
+--- a/drivers/net/atl1c/atl1c_main.c
++++ b/drivers/net/atl1c/atl1c_main.c
+@@ -198,27 +198,12 @@ static void atl1c_phy_config(unsigned long data)
+ 
+ void atl1c_reinit_locked(struct atl1c_adapter *adapter)
+ {
+-
+ 	WARN_ON(in_interrupt());
+ 	atl1c_down(adapter);
+ 	atl1c_up(adapter);
+ 	clear_bit(__AT_RESETTING, &adapter->flags);
+ }
+ 
+-static void atl1c_reset_task(struct work_struct *work)
+-{
+-	struct atl1c_adapter *adapter;
+-	struct net_device *netdev;
+-
+-	adapter = container_of(work, struct atl1c_adapter, reset_task);
+-	netdev = adapter->netdev;
+-
+-	netif_device_detach(netdev);
+-	atl1c_down(adapter);
+-	atl1c_up(adapter);
+-	netif_device_attach(netdev);
+-}
+-
+ static void atl1c_check_link_status(struct atl1c_adapter *adapter)
+ {
+ 	struct atl1c_hw *hw = &adapter->hw;
+@@ -275,18 +260,6 @@ static void atl1c_check_link_status(struct atl1c_adapter *adapter)
+ 	}
+ }
+ 
+-/*
+- * atl1c_link_chg_task - deal with link change event Out of interrupt context
+- * @netdev: network interface device structure
+- */
+-static void atl1c_link_chg_task(struct work_struct *work)
+-{
+-	struct atl1c_adapter *adapter;
+-
+-	adapter = container_of(work, struct atl1c_adapter, link_chg_task);
+-	atl1c_check_link_status(adapter);
+-}
+-
+ static void atl1c_link_chg_event(struct atl1c_adapter *adapter)
+ {
+ 	struct net_device *netdev = adapter->netdev;
+@@ -311,20 +284,40 @@ static void atl1c_link_chg_event(struct atl1c_adapter *adapter)
+ 			adapter->link_speed = SPEED_0;
+ 		}
+ 	}
+-	schedule_work(&adapter->link_chg_task);
++
++	adapter->work_event |= ATL1C_WORK_EVENT_LINK_CHANGE;
++	schedule_work(&adapter->common_task);
+ }
+ 
+-static void atl1c_del_timer(struct atl1c_adapter *adapter)
++static void atl1c_common_task(struct work_struct *work)
+ {
+-	del_timer_sync(&adapter->phy_config_timer);
++	struct atl1c_adapter *adapter;
++	struct net_device *netdev;
++
++	adapter = container_of(work, struct atl1c_adapter, common_task);
++	netdev = adapter->netdev;
++
++	if (adapter->work_event & ATL1C_WORK_EVENT_RESET) {
++		netif_device_detach(netdev);
++		atl1c_down(adapter);
++		atl1c_up(adapter);
++		netif_device_attach(netdev);
++		return;
++	}
++
++	if (adapter->work_event & ATL1C_WORK_EVENT_LINK_CHANGE)
++		atl1c_check_link_status(adapter);
++
++	return;
+ }
+ 
+-static void atl1c_cancel_work(struct atl1c_adapter *adapter)
++
++static void atl1c_del_timer(struct atl1c_adapter *adapter)
+ {
+-	cancel_work_sync(&adapter->reset_task);
+-	cancel_work_sync(&adapter->link_chg_task);
++	del_timer_sync(&adapter->phy_config_timer);
+ }
+ 
++
+ /*
+  * atl1c_tx_timeout - Respond to a Tx Hang
+  * @netdev: network interface device structure
+@@ -334,7 +327,8 @@ static void atl1c_tx_timeout(struct net_device *netdev)
+ 	struct atl1c_adapter *adapter = netdev_priv(netdev);
+ 
+ 	/* Do the reset outside of interrupt context */
+-	schedule_work(&adapter->reset_task);
++	adapter->work_event |= ATL1C_WORK_EVENT_RESET;
++	schedule_work(&adapter->common_task);
+ }
+ 
+ /*
+@@ -1539,7 +1533,8 @@ static irqreturn_t atl1c_intr(int irq, void *data)
+ 			/* reset MAC */
+ 			hw->intr_mask &= ~ISR_ERROR;
+ 			AT_WRITE_REG(hw, REG_IMR, hw->intr_mask);
+-			schedule_work(&adapter->reset_task);
++			adapter->work_event |= ATL1C_WORK_EVENT_RESET;
++			schedule_work(&adapter->common_task);
+ 			break;
+ 		}
+ 
+@@ -2208,8 +2203,7 @@ void atl1c_down(struct atl1c_adapter *adapter)
+ 	struct net_device *netdev = adapter->netdev;
+ 
+ 	atl1c_del_timer(adapter);
+-	atl1c_cancel_work(adapter);
+-
++	adapter->work_event = 0; /* clear all event */
+ 	/* signal that we're down so the interrupt handler does not
+ 	 * reschedule our watchdog timer */
+ 	set_bit(__AT_DOWN, &adapter->flags);
+@@ -2609,8 +2603,8 @@ static int __devinit atl1c_probe(struct pci_dev *pdev,
+ 			adapter->hw.mac_addr[4], adapter->hw.mac_addr[5]);
+ 
+ 	atl1c_hw_set_mac_addr(&adapter->hw);
+-	INIT_WORK(&adapter->reset_task, atl1c_reset_task);
+-	INIT_WORK(&adapter->link_chg_task, atl1c_link_chg_task);
++	INIT_WORK(&adapter->common_task, atl1c_common_task);
++	adapter->work_event = 0;
+ 	err = register_netdev(netdev);
+ 	if (err) {
+ 		dev_err(&pdev->dev, "register netdevice failed\n");
+-- 
+1.6.5.3
+

Modified: dists/trunk/linux-2.6/debian/patches/series/2
==============================================================================
--- dists/trunk/linux-2.6/debian/patches/series/2	Thu Dec 10 01:48:04 2009	(r14768)
+++ dists/trunk/linux-2.6/debian/patches/series/2	Thu Dec 10 03:58:05 2009	(r14769)
@@ -1 +1,2 @@
 + features/all/aufs2/aufs2-20091205.patch
++ bugfix/all/atl1c-use-common_task-instead-of-reset_task-and-link.patch



More information about the Kernel-svn-changes mailing list