diff options
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/net/8139too.c | 25 |
1 files changed, 9 insertions, 16 deletions
diff --git a/drivers/net/8139too.c b/drivers/net/8139too.c index 9de58e249bc3..bcea9d41c32c 100644 --- a/drivers/net/8139too.c +++ b/drivers/net/8139too.c | |||
@@ -586,7 +586,8 @@ struct rtl8139_private { | |||
586 | dma_addr_t tx_bufs_dma; | 586 | dma_addr_t tx_bufs_dma; |
587 | signed char phys[4]; /* MII device addresses. */ | 587 | signed char phys[4]; /* MII device addresses. */ |
588 | char twistie, twist_row, twist_col; /* Twister tune state. */ | 588 | char twistie, twist_row, twist_col; /* Twister tune state. */ |
589 | unsigned int default_port:4; /* Last dev->if_port value. */ | 589 | unsigned int default_port : 4; /* Last dev->if_port value. */ |
590 | unsigned int have_thread : 1; | ||
590 | spinlock_t lock; | 591 | spinlock_t lock; |
591 | spinlock_t rx_lock; | 592 | spinlock_t rx_lock; |
592 | chip_t chipset; | 593 | chip_t chipset; |
@@ -594,7 +595,6 @@ struct rtl8139_private { | |||
594 | struct rtl_extra_stats xstats; | 595 | struct rtl_extra_stats xstats; |
595 | 596 | ||
596 | struct work_struct thread; | 597 | struct work_struct thread; |
597 | long time_to_die; /* -1 no thr, 0 thr active, 1 thr cancel */ | ||
598 | 598 | ||
599 | struct mii_if_info mii; | 599 | struct mii_if_info mii; |
600 | unsigned int regs_len; | 600 | unsigned int regs_len; |
@@ -1599,40 +1599,33 @@ static void rtl8139_thread (void *_data) | |||
1599 | struct net_device *dev = _data; | 1599 | struct net_device *dev = _data; |
1600 | struct rtl8139_private *tp = netdev_priv(dev); | 1600 | struct rtl8139_private *tp = netdev_priv(dev); |
1601 | 1601 | ||
1602 | if ((tp->time_to_die == 0) && | 1602 | if (rtnl_lock_interruptible() == 0) { |
1603 | (rtnl_lock_interruptible() == 0)) { | ||
1604 | rtl8139_thread_iter (dev, tp, tp->mmio_addr); | 1603 | rtl8139_thread_iter (dev, tp, tp->mmio_addr); |
1605 | rtnl_unlock (); | 1604 | rtnl_unlock (); |
1606 | } | 1605 | } |
1607 | 1606 | ||
1608 | if (tp->time_to_die == 0) | 1607 | schedule_delayed_work(&tp->thread, next_tick); |
1609 | schedule_delayed_work(&tp->thread, next_tick); | ||
1610 | } | 1608 | } |
1611 | 1609 | ||
1612 | static void rtl8139_start_thread(struct rtl8139_private *tp) | 1610 | static void rtl8139_start_thread(struct rtl8139_private *tp) |
1613 | { | 1611 | { |
1614 | tp->twistie = 0; | 1612 | tp->twistie = 0; |
1615 | tp->time_to_die = -1; | ||
1616 | if (tp->chipset == CH_8139_K) | 1613 | if (tp->chipset == CH_8139_K) |
1617 | tp->twistie = 1; | 1614 | tp->twistie = 1; |
1618 | else if (tp->drv_flags & HAS_LNK_CHNG) | 1615 | else if (tp->drv_flags & HAS_LNK_CHNG) |
1619 | return; | 1616 | return; |
1620 | 1617 | ||
1621 | tp->time_to_die = 0; | 1618 | tp->have_thread = 1; |
1622 | 1619 | ||
1623 | schedule_delayed_work(&tp->thread, next_tick); | 1620 | schedule_delayed_work(&tp->thread, next_tick); |
1624 | } | 1621 | } |
1625 | 1622 | ||
1626 | static void rtl8139_stop_thread(struct rtl8139_private *tp) | 1623 | static void rtl8139_stop_thread(struct rtl8139_private *tp) |
1627 | { | 1624 | { |
1628 | if (tp->time_to_die < 0) | 1625 | if (tp->have_thread) { |
1629 | return; | 1626 | cancel_rearming_delayed_work(&tp->thread); |
1630 | 1627 | tp->have_thread = 0; | |
1631 | tp->time_to_die = 1; | 1628 | } |
1632 | wmb(); | ||
1633 | |||
1634 | if (cancel_delayed_work(&tp->thread) == 0) | ||
1635 | flush_scheduled_work(); | ||
1636 | } | 1629 | } |
1637 | 1630 | ||
1638 | static inline void rtl8139_tx_clear (struct rtl8139_private *tp) | 1631 | static inline void rtl8139_tx_clear (struct rtl8139_private *tp) |