diff options
Diffstat (limited to 'net/sched/sch_generic.c')
-rw-r--r-- | net/sched/sch_generic.c | 29 |
1 files changed, 7 insertions, 22 deletions
diff --git a/net/sched/sch_generic.c b/net/sched/sch_generic.c index 52eb3439d7c6..1894eb72f6cf 100644 --- a/net/sched/sch_generic.c +++ b/net/sched/sch_generic.c | |||
@@ -36,34 +36,23 @@ | |||
36 | 36 | ||
37 | /* Main transmission queue. */ | 37 | /* Main transmission queue. */ |
38 | 38 | ||
39 | /* Main qdisc structure lock. | 39 | /* Modifications to data participating in scheduling must be protected with |
40 | 40 | * dev->queue_lock spinlock. | |
41 | However, modifications | 41 | * |
42 | to data, participating in scheduling must be additionally | 42 | * The idea is the following: |
43 | protected with dev->queue_lock spinlock. | 43 | * - enqueue, dequeue are serialized via top level device |
44 | 44 | * spinlock dev->queue_lock. | |
45 | The idea is the following: | 45 | * - updates to tree and tree walking are only done under the rtnl mutex. |
46 | - enqueue, dequeue are serialized via top level device | ||
47 | spinlock dev->queue_lock. | ||
48 | - tree walking is protected by read_lock(qdisc_tree_lock) | ||
49 | and this lock is used only in process context. | ||
50 | - updates to tree are made only under rtnl semaphore, | ||
51 | hence this lock may be made without local bh disabling. | ||
52 | |||
53 | qdisc_tree_lock must be grabbed BEFORE dev->queue_lock! | ||
54 | */ | 46 | */ |
55 | DEFINE_RWLOCK(qdisc_tree_lock); | ||
56 | 47 | ||
57 | void qdisc_lock_tree(struct net_device *dev) | 48 | void qdisc_lock_tree(struct net_device *dev) |
58 | { | 49 | { |
59 | write_lock(&qdisc_tree_lock); | ||
60 | spin_lock_bh(&dev->queue_lock); | 50 | spin_lock_bh(&dev->queue_lock); |
61 | } | 51 | } |
62 | 52 | ||
63 | void qdisc_unlock_tree(struct net_device *dev) | 53 | void qdisc_unlock_tree(struct net_device *dev) |
64 | { | 54 | { |
65 | spin_unlock_bh(&dev->queue_lock); | 55 | spin_unlock_bh(&dev->queue_lock); |
66 | write_unlock(&qdisc_tree_lock); | ||
67 | } | 56 | } |
68 | 57 | ||
69 | /* | 58 | /* |
@@ -528,15 +517,11 @@ void dev_activate(struct net_device *dev) | |||
528 | printk(KERN_INFO "%s: activation failed\n", dev->name); | 517 | printk(KERN_INFO "%s: activation failed\n", dev->name); |
529 | return; | 518 | return; |
530 | } | 519 | } |
531 | write_lock(&qdisc_tree_lock); | ||
532 | list_add_tail(&qdisc->list, &dev->qdisc_list); | 520 | list_add_tail(&qdisc->list, &dev->qdisc_list); |
533 | write_unlock(&qdisc_tree_lock); | ||
534 | } else { | 521 | } else { |
535 | qdisc = &noqueue_qdisc; | 522 | qdisc = &noqueue_qdisc; |
536 | } | 523 | } |
537 | write_lock(&qdisc_tree_lock); | ||
538 | dev->qdisc_sleeping = qdisc; | 524 | dev->qdisc_sleeping = qdisc; |
539 | write_unlock(&qdisc_tree_lock); | ||
540 | } | 525 | } |
541 | 526 | ||
542 | if (!netif_carrier_ok(dev)) | 527 | if (!netif_carrier_ok(dev)) |