aboutsummaryrefslogtreecommitdiffstats
path: root/net/core/dev.c
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 /net/core/dev.c
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 'net/core/dev.c')
-rw-r--r--net/core/dev.c12
1 files changed, 5 insertions, 7 deletions
diff --git a/net/core/dev.c b/net/core/dev.c
index 6bfa78c66c25..1b09f1cae46e 100644
--- a/net/core/dev.c
+++ b/net/core/dev.c
@@ -1282,15 +1282,13 @@ int __skb_linearize(struct sk_buff *skb, gfp_t gfp_mask)
1282 1282
1283#define HARD_TX_LOCK(dev, cpu) { \ 1283#define HARD_TX_LOCK(dev, cpu) { \
1284 if ((dev->features & NETIF_F_LLTX) == 0) { \ 1284 if ((dev->features & NETIF_F_LLTX) == 0) { \
1285 spin_lock(&dev->xmit_lock); \ 1285 netif_tx_lock(dev); \
1286 dev->xmit_lock_owner = cpu; \
1287 } \ 1286 } \
1288} 1287}
1289 1288
1290#define HARD_TX_UNLOCK(dev) { \ 1289#define HARD_TX_UNLOCK(dev) { \
1291 if ((dev->features & NETIF_F_LLTX) == 0) { \ 1290 if ((dev->features & NETIF_F_LLTX) == 0) { \
1292 dev->xmit_lock_owner = -1; \ 1291 netif_tx_unlock(dev); \
1293 spin_unlock(&dev->xmit_lock); \
1294 } \ 1292 } \
1295} 1293}
1296 1294
@@ -1389,8 +1387,8 @@ int dev_queue_xmit(struct sk_buff *skb)
1389 /* The device has no queue. Common case for software devices: 1387 /* The device has no queue. Common case for software devices:
1390 loopback, all the sorts of tunnels... 1388 loopback, all the sorts of tunnels...
1391 1389
1392 Really, it is unlikely that xmit_lock protection is necessary here. 1390 Really, it is unlikely that netif_tx_lock protection is necessary
1393 (f.e. loopback and IP tunnels are clean ignoring statistics 1391 here. (f.e. loopback and IP tunnels are clean ignoring statistics
1394 counters.) 1392 counters.)
1395 However, it is possible, that they rely on protection 1393 However, it is possible, that they rely on protection
1396 made by us here. 1394 made by us here.
@@ -2805,7 +2803,7 @@ int register_netdevice(struct net_device *dev)
2805 BUG_ON(dev->reg_state != NETREG_UNINITIALIZED); 2803 BUG_ON(dev->reg_state != NETREG_UNINITIALIZED);
2806 2804
2807 spin_lock_init(&dev->queue_lock); 2805 spin_lock_init(&dev->queue_lock);
2808 spin_lock_init(&dev->xmit_lock); 2806 spin_lock_init(&dev->_xmit_lock);
2809 dev->xmit_lock_owner = -1; 2807 dev->xmit_lock_owner = -1;
2810#ifdef CONFIG_NET_CLS_ACT 2808#ifdef CONFIG_NET_CLS_ACT
2811 spin_lock_init(&dev->ingress_lock); 2809 spin_lock_init(&dev->ingress_lock);