diff options
author | David S. Miller <davem@davemloft.net> | 2008-07-19 01:50:15 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2008-07-19 01:50:15 -0400 |
commit | 3072367300aa8c779e3a14ee8e89de079e90f3ad (patch) | |
tree | 7f74c5b8fdb300532fbbc83ba00d6d1d17af020e /net/sched/sch_generic.c | |
parent | 72b25a913ed9b1ab49c7022adaf3f271a65ea219 (diff) |
pkt_sched: Manage qdisc list inside of root qdisc.
Idea is from Patrick McHardy.
Instead of managing the list of qdiscs on the device level, manage it
in the root qdisc of a netdev_queue. This solves all kinds of
visibility issues during qdisc destruction.
The way to iterate over all qdiscs of a netdev_queue is to visit
the netdev_queue->qdisc, and then traverse it's list.
The only special case is to ignore builting qdiscs at the root when
dumping or doing a qdisc_lookup(). That was not needed previously
because builtin qdiscs were not added to the device's qdisc_list.
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/sched/sch_generic.c')
-rw-r--r-- | net/sched/sch_generic.c | 10 |
1 files changed, 2 insertions, 8 deletions
diff --git a/net/sched/sch_generic.c b/net/sched/sch_generic.c index e244c462e6bd..14cc443d0490 100644 --- a/net/sched/sch_generic.c +++ b/net/sched/sch_generic.c | |||
@@ -480,15 +480,12 @@ static void __qdisc_destroy(struct rcu_head *head) | |||
480 | 480 | ||
481 | void qdisc_destroy(struct Qdisc *qdisc) | 481 | void qdisc_destroy(struct Qdisc *qdisc) |
482 | { | 482 | { |
483 | struct net_device *dev = qdisc_dev(qdisc); | ||
484 | |||
485 | if (qdisc->flags & TCQ_F_BUILTIN || | 483 | if (qdisc->flags & TCQ_F_BUILTIN || |
486 | !atomic_dec_and_test(&qdisc->refcnt)) | 484 | !atomic_dec_and_test(&qdisc->refcnt)) |
487 | return; | 485 | return; |
488 | 486 | ||
489 | spin_lock_bh(&dev->qdisc_list_lock); | 487 | if (qdisc->parent) |
490 | list_del(&qdisc->list); | 488 | list_del(&qdisc->list); |
491 | spin_unlock_bh(&dev->qdisc_list_lock); | ||
492 | 489 | ||
493 | call_rcu(&qdisc->q_rcu, __qdisc_destroy); | 490 | call_rcu(&qdisc->q_rcu, __qdisc_destroy); |
494 | } | 491 | } |
@@ -520,9 +517,6 @@ static void attach_one_default_qdisc(struct net_device *dev, | |||
520 | printk(KERN_INFO "%s: activation failed\n", dev->name); | 517 | printk(KERN_INFO "%s: activation failed\n", dev->name); |
521 | return; | 518 | return; |
522 | } | 519 | } |
523 | spin_lock_bh(&dev->qdisc_list_lock); | ||
524 | list_add_tail(&qdisc->list, &dev->qdisc_list); | ||
525 | spin_unlock_bh(&dev->qdisc_list_lock); | ||
526 | } else { | 520 | } else { |
527 | qdisc = &noqueue_qdisc; | 521 | qdisc = &noqueue_qdisc; |
528 | } | 522 | } |