aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEric Dumazet <edumazet@google.com>2013-12-12 18:41:56 -0500
committerDavid S. Miller <davem@davemloft.net>2013-12-14 01:20:06 -0500
commite57a784d8cae429f5b697fe55abf420181d9ff09 (patch)
tree69c247ac88bc588611f280bf684ae0a707523c63
parent59bcaed5f71f002d39d151ccd387dd0f226e00f2 (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>
-rw-r--r--net/sched/sch_api.c5
-rw-r--r--net/sched/sch_generic.c2
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
274void qdisc_list_add(struct Qdisc *q) 274void 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}
279EXPORT_SYMBOL(qdisc_list_add); 282EXPORT_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}