diff options
author | Eric Dumazet <edumazet@google.com> | 2013-12-12 18:41:56 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-12-14 01:20:06 -0500 |
commit | e57a784d8cae429f5b697fe55abf420181d9ff09 (patch) | |
tree | 69c247ac88bc588611f280bf684ae0a707523c63 /net | |
parent | 59bcaed5f71f002d39d151ccd387dd0f226e00f2 (diff) |
pkt_sched: set root qdisc before change() in attach_default_qdiscs()
After commit 95dc19299f74 ("pkt_sched: give visibility to mq slave
qdiscs") we call disc_list_add() while the device qdisc might be
the noop_qdisc one.
This shows up as duplicates in "tc qdisc show", as all inactive devices
point to noop_qdisc.
Fix this by setting dev->qdisc to the new qdisc before calling
ops->change() in attach_default_qdiscs()
Add a WARN_ON_ONCE() to catch any future similar problem.
Signed-off-by: Eric Dumazet <edumazet@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net')
-rw-r--r-- | net/sched/sch_api.c | 5 | ||||
-rw-r--r-- | net/sched/sch_generic.c | 2 |
2 files changed, 5 insertions, 2 deletions
diff --git a/net/sched/sch_api.c b/net/sched/sch_api.c index 547b4a88ae2a..c31190e29b90 100644 --- a/net/sched/sch_api.c +++ b/net/sched/sch_api.c | |||
@@ -273,8 +273,11 @@ static struct Qdisc *qdisc_match_from_root(struct Qdisc *root, u32 handle) | |||
273 | 273 | ||
274 | void qdisc_list_add(struct Qdisc *q) | 274 | void qdisc_list_add(struct Qdisc *q) |
275 | { | 275 | { |
276 | struct Qdisc *root = qdisc_dev(q)->qdisc; | ||
277 | |||
278 | WARN_ON_ONCE(root == &noop_qdisc); | ||
276 | if ((q->parent != TC_H_ROOT) && !(q->flags & TCQ_F_INGRESS)) | 279 | if ((q->parent != TC_H_ROOT) && !(q->flags & TCQ_F_INGRESS)) |
277 | list_add_tail(&q->list, &qdisc_dev(q)->qdisc->list); | 280 | list_add_tail(&q->list, &root->list); |
278 | } | 281 | } |
279 | EXPORT_SYMBOL(qdisc_list_add); | 282 | EXPORT_SYMBOL(qdisc_list_add); |
280 | 283 | ||
diff --git a/net/sched/sch_generic.c b/net/sched/sch_generic.c index 6a91d7d48ade..32bb942d2faa 100644 --- a/net/sched/sch_generic.c +++ b/net/sched/sch_generic.c | |||
@@ -718,8 +718,8 @@ static void attach_default_qdiscs(struct net_device *dev) | |||
718 | } else { | 718 | } else { |
719 | qdisc = qdisc_create_dflt(txq, &mq_qdisc_ops, TC_H_ROOT); | 719 | qdisc = qdisc_create_dflt(txq, &mq_qdisc_ops, TC_H_ROOT); |
720 | if (qdisc) { | 720 | if (qdisc) { |
721 | qdisc->ops->attach(qdisc); | ||
722 | dev->qdisc = qdisc; | 721 | dev->qdisc = qdisc; |
722 | qdisc->ops->attach(qdisc); | ||
723 | } | 723 | } |
724 | } | 724 | } |
725 | } | 725 | } |