[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20081124105023.GA13957@ff.dom.local>
Date: Mon, 24 Nov 2008 10:50:23 +0000
From: Jarek Poplawski <jarkao2@...il.com>
To: Patrick McHardy <kaber@...sh.net>
Cc: "David S. Miller" <davem@...emloft.net>,
Linux Netdev List <netdev@...r.kernel.org>
Subject: [PATCH] pkt_sched: sch_drr: Fix drr_dequeue() loop
pkt_sched: sch_drr: Fix loop in drr_dequeue
If all child qdiscs of sch_drr are non-work-conserving (e.g. sch_tbf)
drr_dequeue() will busy-loop waiting for skbs instead of leaving the
job for a watchdog. Checking for list_empty() in each loop isn't
necessary either, because this can never be true exept the first time.
Signed-off-by: Jarek Poplawski <jarkao2@...il.com>
---
net/sched/sch_drr.c | 24 +++++++++++++++++++++---
1 files changed, 21 insertions(+), 3 deletions(-)
diff --git a/net/sched/sch_drr.c b/net/sched/sch_drr.c
index 37e6ab9..ab75461 100644
--- a/net/sched/sch_drr.c
+++ b/net/sched/sch_drr.c
@@ -369,12 +369,17 @@ static int drr_enqueue(struct sk_buff *skb, struct Qdisc *sch)
static struct sk_buff *drr_dequeue(struct Qdisc *sch)
{
struct drr_sched *q = qdisc_priv(sch);
- struct drr_class *cl;
+ struct drr_class *cl, *cl_first;
struct sk_buff *skb;
unsigned int len;
+ bool skb_waiting = false;
- while (!list_empty(&q->active)) {
- cl = list_first_entry(&q->active, struct drr_class, alist);
+ if (list_empty(&q->active))
+ return NULL;
+
+ cl_first = cl = list_first_entry(&q->active, struct drr_class, alist);
+
+ while (1) {
skb = cl->qdisc->ops->peek(cl->qdisc);
if (skb == NULL)
goto skip;
@@ -390,9 +395,22 @@ static struct sk_buff *drr_dequeue(struct Qdisc *sch)
}
cl->deficit += cl->quantum;
+ skb_waiting = true;
skip:
list_move_tail(&cl->alist, &q->active);
+ cl = list_first_entry(&q->active, struct drr_class, alist);
+
+ if (cl == cl_first) {
+ if (skb_waiting) {
+ /* next round of deficit refilling */
+ skb_waiting = false;
+ } else {
+ /* all qdiscs are non-work-conserving! */
+ break;
+ }
+ }
}
+
return NULL;
}
--
To unsubscribe from this list: send the line "unsubscribe netdev" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Powered by blists - more mailing lists