aboutsummaryrefslogtreecommitdiffstats
path: root/include/linux/netdevice.h
diff options
context:
space:
mode:
authorHerbert Xu <herbert@gondor.apana.org.au>2006-06-09 15:20:56 -0400
committerDavid S. Miller <davem@sunset.davemloft.net>2006-06-18 00:30:14 -0400
commit932ff279a43ab7257942cddff2595acd541cc49b (patch)
treee60130673a20d71becdac858c2589d8dfbf3ae1f /include/linux/netdevice.h
parentbf0857ea32addb6bc8b46383604b218b8ec09f19 (diff)
[NET]: Add netif_tx_lock
Various drivers use xmit_lock internally to synchronise with their transmission routines. They do so without setting xmit_lock_owner. This is fine as long as netpoll is not in use. With netpoll it is possible for deadlocks to occur if xmit_lock_owner isn't set. This is because if a printk occurs while xmit_lock is held and xmit_lock_owner is not set can cause netpoll to attempt to take xmit_lock recursively. While it is possible to resolve this by getting netpoll to use trylock, it is suboptimal because netpoll's sole objective is to maximise the chance of getting the printk out on the wire. So delaying or dropping the message is to be avoided as much as possible. So the only alternative is to always set xmit_lock_owner. The following patch does this by introducing the netif_tx_lock family of functions that take care of setting/unsetting xmit_lock_owner. I renamed xmit_lock to _xmit_lock to indicate that it should not be used directly. I didn't provide irq versions of the netif_tx_lock functions since xmit_lock is meant to be a BH-disabling lock. This is pretty much a straight text substitution except for a small bug fix in winbond. It currently uses netif_stop_queue/spin_unlock_wait to stop transmission. This is unsafe as an IRQ can potentially wake up the queue. So it is safer to use netif_tx_disable. The hamradio bits used spin_lock_irq but it is unnecessary as xmit_lock must never be taken in an IRQ handler. Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'include/linux/netdevice.h')
-rw-r--r--include/linux/netdevice.h38
1 files changed, 35 insertions, 3 deletions
diff --git a/include/linux/netdevice.h b/include/linux/netdevice.h
index b5760c67af9..067b9ccafd8 100644
--- a/include/linux/netdevice.h
+++ b/include/linux/netdevice.h
@@ -407,7 +407,7 @@ struct net_device
407 * One part is mostly used on xmit path (device) 407 * One part is mostly used on xmit path (device)
408 */ 408 */
409 /* hard_start_xmit synchronizer */ 409 /* hard_start_xmit synchronizer */
410 spinlock_t xmit_lock ____cacheline_aligned_in_smp; 410 spinlock_t _xmit_lock ____cacheline_aligned_in_smp;
411 /* cpu id of processor entered to hard_start_xmit or -1, 411 /* cpu id of processor entered to hard_start_xmit or -1,
412 if nobody entered there. 412 if nobody entered there.
413 */ 413 */
@@ -893,11 +893,43 @@ static inline void __netif_rx_complete(struct net_device *dev)
893 clear_bit(__LINK_STATE_RX_SCHED, &dev->state); 893 clear_bit(__LINK_STATE_RX_SCHED, &dev->state);
894} 894}
895 895
896static inline void netif_tx_lock(struct net_device *dev)
897{
898 spin_lock(&dev->_xmit_lock);
899 dev->xmit_lock_owner = smp_processor_id();
900}
901
902static inline void netif_tx_lock_bh(struct net_device *dev)
903{
904 spin_lock_bh(&dev->_xmit_lock);
905 dev->xmit_lock_owner = smp_processor_id();
906}
907
908static inline int netif_tx_trylock(struct net_device *dev)
909{
910 int err = spin_trylock(&dev->_xmit_lock);
911 if (!err)
912 dev->xmit_lock_owner = smp_processor_id();
913 return err;
914}
915
916static inline void netif_tx_unlock(struct net_device *dev)
917{
918 dev->xmit_lock_owner = -1;
919 spin_unlock(&dev->_xmit_lock);
920}
921
922static inline void netif_tx_unlock_bh(struct net_device *dev)
923{
924 dev->xmit_lock_owner = -1;
925 spin_unlock_bh(&dev->_xmit_lock);
926}
927
896static inline void netif_tx_disable(struct net_device *dev) 928static inline void netif_tx_disable(struct net_device *dev)
897{ 929{
898 spin_lock_bh(&dev->xmit_lock); 930 netif_tx_lock_bh(dev);
899 netif_stop_queue(dev); 931 netif_stop_queue(dev);
900 spin_unlock_bh(&dev->xmit_lock); 932 netif_tx_unlock_bh(dev);
901} 933}
902 934
903/* These functions live elsewhere (drivers/net/net_init.c, but related) */ 935/* These functions live elsewhere (drivers/net/net_init.c, but related) */