提交 95946658 编写于 作者: K Konstantin Khlebnikov 提交者: David S. Miller

net_sched: call qlen_notify only if child qdisc is empty

This callback is used for deactivating class in parent qdisc.
This is cheaper to test queue length right here.

Also this allows to catch draining screwed backlog and prevent
second deactivation of already inactive parent class which will
crash kernel for sure. Kernel with print warning at destruction
of child qdisc where no packets but backlog is not zero.
Signed-off-by: NKonstantin Khlebnikov <khlebnikov@yandex-team.ru>
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
上级 869cec99
...@@ -749,6 +749,7 @@ void qdisc_tree_reduce_backlog(struct Qdisc *sch, unsigned int n, ...@@ -749,6 +749,7 @@ void qdisc_tree_reduce_backlog(struct Qdisc *sch, unsigned int n,
const struct Qdisc_class_ops *cops; const struct Qdisc_class_ops *cops;
unsigned long cl; unsigned long cl;
u32 parentid; u32 parentid;
bool notify;
int drops; int drops;
if (n == 0 && len == 0) if (n == 0 && len == 0)
...@@ -761,6 +762,13 @@ void qdisc_tree_reduce_backlog(struct Qdisc *sch, unsigned int n, ...@@ -761,6 +762,13 @@ void qdisc_tree_reduce_backlog(struct Qdisc *sch, unsigned int n,
if (sch->flags & TCQ_F_NOPARENT) if (sch->flags & TCQ_F_NOPARENT)
break; break;
/* Notify parent qdisc only if child qdisc becomes empty.
*
* If child was empty even before update then backlog
* counter is screwed and we skip notification because
* parent class is already passive.
*/
notify = !sch->q.qlen && !WARN_ON_ONCE(!n);
/* TODO: perform the search on a per txq basis */ /* TODO: perform the search on a per txq basis */
sch = qdisc_lookup(qdisc_dev(sch), TC_H_MAJ(parentid)); sch = qdisc_lookup(qdisc_dev(sch), TC_H_MAJ(parentid));
if (sch == NULL) { if (sch == NULL) {
...@@ -768,7 +776,7 @@ void qdisc_tree_reduce_backlog(struct Qdisc *sch, unsigned int n, ...@@ -768,7 +776,7 @@ void qdisc_tree_reduce_backlog(struct Qdisc *sch, unsigned int n,
break; break;
} }
cops = sch->ops->cl_ops; cops = sch->ops->cl_ops;
if (cops->qlen_notify) { if (notify && cops->qlen_notify) {
cl = cops->get(sch, parentid); cl = cops->get(sch, parentid);
cops->qlen_notify(sch, cl); cops->qlen_notify(sch, cl);
cops->put(sch, cl); cops->put(sch, cl);
......
...@@ -1385,7 +1385,6 @@ static void cbq_qlen_notify(struct Qdisc *sch, unsigned long arg) ...@@ -1385,7 +1385,6 @@ static void cbq_qlen_notify(struct Qdisc *sch, unsigned long arg)
{ {
struct cbq_class *cl = (struct cbq_class *)arg; struct cbq_class *cl = (struct cbq_class *)arg;
if (cl->q->q.qlen == 0)
cbq_deactivate_class(cl); cbq_deactivate_class(cl);
} }
......
...@@ -246,7 +246,6 @@ static void drr_qlen_notify(struct Qdisc *csh, unsigned long arg) ...@@ -246,7 +246,6 @@ static void drr_qlen_notify(struct Qdisc *csh, unsigned long arg)
{ {
struct drr_class *cl = (struct drr_class *)arg; struct drr_class *cl = (struct drr_class *)arg;
if (cl->qdisc->q.qlen == 0)
list_del(&cl->alist); list_del(&cl->alist);
} }
......
...@@ -1221,10 +1221,8 @@ hfsc_qlen_notify(struct Qdisc *sch, unsigned long arg) ...@@ -1221,10 +1221,8 @@ hfsc_qlen_notify(struct Qdisc *sch, unsigned long arg)
{ {
struct hfsc_class *cl = (struct hfsc_class *)arg; struct hfsc_class *cl = (struct hfsc_class *)arg;
if (cl->qdisc->q.qlen == 0) {
update_vf(cl, 0, 0); update_vf(cl, 0, 0);
set_passive(cl); set_passive(cl);
}
} }
static unsigned long static unsigned long
......
...@@ -1186,7 +1186,6 @@ static void htb_qlen_notify(struct Qdisc *sch, unsigned long arg) ...@@ -1186,7 +1186,6 @@ static void htb_qlen_notify(struct Qdisc *sch, unsigned long arg)
{ {
struct htb_class *cl = (struct htb_class *)arg; struct htb_class *cl = (struct htb_class *)arg;
if (cl->un.leaf.q->q.qlen == 0)
htb_deactivate(qdisc_priv(sch), cl); htb_deactivate(qdisc_priv(sch), cl);
} }
......
...@@ -1428,7 +1428,6 @@ static void qfq_qlen_notify(struct Qdisc *sch, unsigned long arg) ...@@ -1428,7 +1428,6 @@ static void qfq_qlen_notify(struct Qdisc *sch, unsigned long arg)
struct qfq_sched *q = qdisc_priv(sch); struct qfq_sched *q = qdisc_priv(sch);
struct qfq_class *cl = (struct qfq_class *)arg; struct qfq_class *cl = (struct qfq_class *)arg;
if (cl->qdisc->q.qlen == 0)
qfq_deactivate_class(q, cl); qfq_deactivate_class(q, cl);
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册