aboutsummaryrefslogtreecommitdiffstats
path: root/net/sched
diff options
context:
space:
mode:
Diffstat (limited to 'net/sched')
-rw-r--r--net/sched/sch_netem.c54
1 files changed, 42 insertions, 12 deletions
diff --git a/net/sched/sch_netem.c b/net/sched/sch_netem.c
index 48360f7eec5d..bb9bf8d5003c 100644
--- a/net/sched/sch_netem.c
+++ b/net/sched/sch_netem.c
@@ -62,11 +62,12 @@ struct netem_sched_data {
62 u32 gap; 62 u32 gap;
63 u32 jitter; 63 u32 jitter;
64 u32 duplicate; 64 u32 duplicate;
65 u32 reorder;
65 66
66 struct crndstate { 67 struct crndstate {
67 unsigned long last; 68 unsigned long last;
68 unsigned long rho; 69 unsigned long rho;
69 } delay_cor, loss_cor, dup_cor; 70 } delay_cor, loss_cor, dup_cor, reorder_cor;
70 71
71 struct disttable { 72 struct disttable {
72 u32 size; 73 u32 size;
@@ -180,23 +181,23 @@ static int netem_enqueue(struct sk_buff *skb, struct Qdisc *sch)
180 q->duplicate = dupsave; 181 q->duplicate = dupsave;
181 } 182 }
182 183
183 /* 184 if (q->gap == 0 /* not doing reordering */
184 * Do re-ordering by putting one out of N packets at the front 185 || q->counter < q->gap /* inside last reordering gap */
185 * of the queue. 186 || q->reorder < get_crandom(&q->reorder_cor)) {
186 * gap == 0 is special case for no-reordering.
187 */
188 if (q->gap == 0 || q->counter != q->gap) {
189 psched_time_t now; 187 psched_time_t now;
190 PSCHED_GET_TIME(now); 188 PSCHED_GET_TIME(now);
191 PSCHED_TADD2(now, 189 PSCHED_TADD2(now, tabledist(q->latency, q->jitter,
192 tabledist(q->latency, q->jitter, &q->delay_cor, q->delay_dist), 190 &q->delay_cor, q->delay_dist),
193 cb->time_to_send); 191 cb->time_to_send);
194
195 ++q->counter; 192 ++q->counter;
196 ret = q->qdisc->enqueue(skb, q->qdisc); 193 ret = q->qdisc->enqueue(skb, q->qdisc);
197 } else { 194 } else {
198 q->counter = 0; 195 /*
196 * Do re-ordering by putting one out of N packets at the front
197 * of the queue.
198 */
199 PSCHED_GET_TIME(cb->time_to_send); 199 PSCHED_GET_TIME(cb->time_to_send);
200 q->counter = 0;
200 ret = q->qdisc->ops->requeue(skb, q->qdisc); 201 ret = q->qdisc->ops->requeue(skb, q->qdisc);
201 } 202 }
202 203
@@ -351,6 +352,19 @@ static int get_correlation(struct Qdisc *sch, const struct rtattr *attr)
351 return 0; 352 return 0;
352} 353}
353 354
355static int get_reorder(struct Qdisc *sch, const struct rtattr *attr)
356{
357 struct netem_sched_data *q = qdisc_priv(sch);
358 const struct tc_netem_reorder *r = RTA_DATA(attr);
359
360 if (RTA_PAYLOAD(attr) != sizeof(*r))
361 return -EINVAL;
362
363 q->reorder = r->probability;
364 init_crandom(&q->reorder_cor, r->correlation);
365 return 0;
366}
367
354static int netem_change(struct Qdisc *sch, struct rtattr *opt) 368static int netem_change(struct Qdisc *sch, struct rtattr *opt)
355{ 369{
356 struct netem_sched_data *q = qdisc_priv(sch); 370 struct netem_sched_data *q = qdisc_priv(sch);
@@ -371,9 +385,15 @@ static int netem_change(struct Qdisc *sch, struct rtattr *opt)
371 q->jitter = qopt->jitter; 385 q->jitter = qopt->jitter;
372 q->limit = qopt->limit; 386 q->limit = qopt->limit;
373 q->gap = qopt->gap; 387 q->gap = qopt->gap;
388 q->counter = 0;
374 q->loss = qopt->loss; 389 q->loss = qopt->loss;
375 q->duplicate = qopt->duplicate; 390 q->duplicate = qopt->duplicate;
376 391
392 /* for compatiablity with earlier versions.
393 * if gap is set, need to assume 100% probablity
394 */
395 q->reorder = ~0;
396
377 /* Handle nested options after initial queue options. 397 /* Handle nested options after initial queue options.
378 * Should have put all options in nested format but too late now. 398 * Should have put all options in nested format but too late now.
379 */ 399 */
@@ -395,6 +415,11 @@ static int netem_change(struct Qdisc *sch, struct rtattr *opt)
395 if (ret) 415 if (ret)
396 return ret; 416 return ret;
397 } 417 }
418 if (tb[TCA_NETEM_REORDER-1]) {
419 ret = get_reorder(sch, tb[TCA_NETEM_REORDER-1]);
420 if (ret)
421 return ret;
422 }
398 } 423 }
399 424
400 425
@@ -412,7 +437,6 @@ static int netem_init(struct Qdisc *sch, struct rtattr *opt)
412 init_timer(&q->timer); 437 init_timer(&q->timer);
413 q->timer.function = netem_watchdog; 438 q->timer.function = netem_watchdog;
414 q->timer.data = (unsigned long) sch; 439 q->timer.data = (unsigned long) sch;
415 q->counter = 0;
416 440
417 q->qdisc = qdisc_create_dflt(sch->dev, &pfifo_qdisc_ops); 441 q->qdisc = qdisc_create_dflt(sch->dev, &pfifo_qdisc_ops);
418 if (!q->qdisc) { 442 if (!q->qdisc) {
@@ -444,6 +468,7 @@ static int netem_dump(struct Qdisc *sch, struct sk_buff *skb)
444 struct rtattr *rta = (struct rtattr *) b; 468 struct rtattr *rta = (struct rtattr *) b;
445 struct tc_netem_qopt qopt; 469 struct tc_netem_qopt qopt;
446 struct tc_netem_corr cor; 470 struct tc_netem_corr cor;
471 struct tc_netem_reorder reorder;
447 472
448 qopt.latency = q->latency; 473 qopt.latency = q->latency;
449 qopt.jitter = q->jitter; 474 qopt.jitter = q->jitter;
@@ -457,6 +482,11 @@ static int netem_dump(struct Qdisc *sch, struct sk_buff *skb)
457 cor.loss_corr = q->loss_cor.rho; 482 cor.loss_corr = q->loss_cor.rho;
458 cor.dup_corr = q->dup_cor.rho; 483 cor.dup_corr = q->dup_cor.rho;
459 RTA_PUT(skb, TCA_NETEM_CORR, sizeof(cor), &cor); 484 RTA_PUT(skb, TCA_NETEM_CORR, sizeof(cor), &cor);
485
486 reorder.probability = q->reorder;
487 reorder.correlation = q->reorder_cor.rho;
488 RTA_PUT(skb, TCA_NETEM_REORDER, sizeof(reorder), &reorder);
489
460 rta->rta_len = skb->tail - b; 490 rta->rta_len = skb->tail - b;
461 491
462 return skb->len; 492 return skb->len;