diff options
author | Eric Dumazet <edumazet@google.com> | 2013-06-15 06:30:10 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-06-20 02:06:52 -0400 |
commit | c9364636dcb01a6fc37ca2c6a51c5aa0c663013c (patch) | |
tree | 4ea94902a65b0f3ca46ec82652da39b0dea59d55 | |
parent | bcefe17cffd06efdda3e7ad679ea743236e6271a (diff) |
htb: refactor struct htb_sched fields for performance
htb_sched structures are big, and source of false sharing on SMP.
Every time a packet is queued or dequeue, many cache lines must be
touched because structures are not lay out properly.
By carefully splitting htb_sched in two parts, and define sub structures
to increase data locality, we can improve performance dramatically on
SMP.
New htb_prio structure can also be used in htb_class to increase data
locality.
I got 26 % performance increase on a 24 threads machine, with 200
concurrent netperf in TCP_RR mode, using a HTB hierarchy of 4 classes.
Signed-off-by: Eric Dumazet <edumazet@google.com>
Cc: Tom Herbert <therbert@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | net/sched/sch_htb.c | 181 |
1 files changed, 95 insertions, 86 deletions
diff --git a/net/sched/sch_htb.c b/net/sched/sch_htb.c index 7954e73d118a..c2124ea29f45 100644 --- a/net/sched/sch_htb.c +++ b/net/sched/sch_htb.c | |||
@@ -76,6 +76,20 @@ enum htb_cmode { | |||
76 | HTB_CAN_SEND /* class can send */ | 76 | HTB_CAN_SEND /* class can send */ |
77 | }; | 77 | }; |
78 | 78 | ||
79 | struct htb_prio { | ||
80 | union { | ||
81 | struct rb_root row; | ||
82 | struct rb_root feed; | ||
83 | }; | ||
84 | struct rb_node *ptr; | ||
85 | /* When class changes from state 1->2 and disconnects from | ||
86 | * parent's feed then we lost ptr value and start from the | ||
87 | * first child again. Here we store classid of the | ||
88 | * last valid ptr (used when ptr is NULL). | ||
89 | */ | ||
90 | u32 last_ptr_id; | ||
91 | }; | ||
92 | |||
79 | /* interior & leaf nodes; props specific to leaves are marked L: | 93 | /* interior & leaf nodes; props specific to leaves are marked L: |
80 | * To reduce false sharing, place mostly read fields at beginning, | 94 | * To reduce false sharing, place mostly read fields at beginning, |
81 | * and mostly written ones at the end. | 95 | * and mostly written ones at the end. |
@@ -112,19 +126,12 @@ struct htb_class { | |||
112 | 126 | ||
113 | union { | 127 | union { |
114 | struct htb_class_leaf { | 128 | struct htb_class_leaf { |
115 | struct Qdisc *q; | ||
116 | int deficit[TC_HTB_MAXDEPTH]; | ||
117 | struct list_head drop_list; | 129 | struct list_head drop_list; |
130 | int deficit[TC_HTB_MAXDEPTH]; | ||
131 | struct Qdisc *q; | ||
118 | } leaf; | 132 | } leaf; |
119 | struct htb_class_inner { | 133 | struct htb_class_inner { |
120 | struct rb_root feed[TC_HTB_NUMPRIO]; /* feed trees */ | 134 | struct htb_prio clprio[TC_HTB_NUMPRIO]; |
121 | struct rb_node *ptr[TC_HTB_NUMPRIO]; /* current class ptr */ | ||
122 | /* When class changes from state 1->2 and disconnects from | ||
123 | * parent's feed then we lost ptr value and start from the | ||
124 | * first child again. Here we store classid of the | ||
125 | * last valid ptr (used when ptr is NULL). | ||
126 | */ | ||
127 | u32 last_ptr_id[TC_HTB_NUMPRIO]; | ||
128 | } inner; | 135 | } inner; |
129 | } un; | 136 | } un; |
130 | s64 pq_key; | 137 | s64 pq_key; |
@@ -135,40 +142,39 @@ struct htb_class { | |||
135 | struct rb_node node[TC_HTB_NUMPRIO]; /* node for self or feed tree */ | 142 | struct rb_node node[TC_HTB_NUMPRIO]; /* node for self or feed tree */ |
136 | }; | 143 | }; |
137 | 144 | ||
145 | struct htb_level { | ||
146 | struct rb_root wait_pq; | ||
147 | struct htb_prio hprio[TC_HTB_NUMPRIO]; | ||
148 | }; | ||
149 | |||
138 | struct htb_sched { | 150 | struct htb_sched { |
139 | struct Qdisc_class_hash clhash; | 151 | struct Qdisc_class_hash clhash; |
140 | struct list_head drops[TC_HTB_NUMPRIO];/* active leaves (for drops) */ | 152 | int defcls; /* class where unclassified flows go to */ |
141 | 153 | int rate2quantum; /* quant = rate / rate2quantum */ | |
142 | /* self list - roots of self generating tree */ | ||
143 | struct rb_root row[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO]; | ||
144 | int row_mask[TC_HTB_MAXDEPTH]; | ||
145 | struct rb_node *ptr[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO]; | ||
146 | u32 last_ptr_id[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO]; | ||
147 | 154 | ||
148 | /* self wait list - roots of wait PQs per row */ | 155 | /* filters for qdisc itself */ |
149 | struct rb_root wait_pq[TC_HTB_MAXDEPTH]; | 156 | struct tcf_proto *filter_list; |
150 | 157 | ||
151 | /* time of nearest event per level (row) */ | 158 | #define HTB_WARN_TOOMANYEVENTS 0x1 |
152 | s64 near_ev_cache[TC_HTB_MAXDEPTH]; | 159 | unsigned int warned; /* only one warning */ |
160 | int direct_qlen; | ||
161 | struct work_struct work; | ||
153 | 162 | ||
154 | int defcls; /* class where unclassified flows go to */ | 163 | /* non shaped skbs; let them go directly thru */ |
164 | struct sk_buff_head direct_queue; | ||
165 | long direct_pkts; | ||
155 | 166 | ||
156 | /* filters for qdisc itself */ | 167 | struct qdisc_watchdog watchdog; |
157 | struct tcf_proto *filter_list; | ||
158 | 168 | ||
159 | int rate2quantum; /* quant = rate / rate2quantum */ | 169 | s64 now; /* cached dequeue time */ |
160 | s64 now; /* cached dequeue time */ | 170 | struct list_head drops[TC_HTB_NUMPRIO];/* active leaves (for drops) */ |
161 | struct qdisc_watchdog watchdog; | ||
162 | 171 | ||
163 | /* non shaped skbs; let them go directly thru */ | 172 | /* time of nearest event per level (row) */ |
164 | struct sk_buff_head direct_queue; | 173 | s64 near_ev_cache[TC_HTB_MAXDEPTH]; |
165 | int direct_qlen; /* max qlen of above */ | ||
166 | 174 | ||
167 | long direct_pkts; | 175 | int row_mask[TC_HTB_MAXDEPTH]; |
168 | 176 | ||
169 | #define HTB_WARN_TOOMANYEVENTS 0x1 | 177 | struct htb_level hlevel[TC_HTB_MAXDEPTH]; |
170 | unsigned int warned; /* only one warning */ | ||
171 | struct work_struct work; | ||
172 | }; | 178 | }; |
173 | 179 | ||
174 | /* find class in global hash table using given handle */ | 180 | /* find class in global hash table using given handle */ |
@@ -284,7 +290,7 @@ static void htb_add_to_id_tree(struct rb_root *root, | |||
284 | static void htb_add_to_wait_tree(struct htb_sched *q, | 290 | static void htb_add_to_wait_tree(struct htb_sched *q, |
285 | struct htb_class *cl, s64 delay) | 291 | struct htb_class *cl, s64 delay) |
286 | { | 292 | { |
287 | struct rb_node **p = &q->wait_pq[cl->level].rb_node, *parent = NULL; | 293 | struct rb_node **p = &q->hlevel[cl->level].wait_pq.rb_node, *parent = NULL; |
288 | 294 | ||
289 | cl->pq_key = q->now + delay; | 295 | cl->pq_key = q->now + delay; |
290 | if (cl->pq_key == q->now) | 296 | if (cl->pq_key == q->now) |
@@ -304,7 +310,7 @@ static void htb_add_to_wait_tree(struct htb_sched *q, | |||
304 | p = &parent->rb_left; | 310 | p = &parent->rb_left; |
305 | } | 311 | } |
306 | rb_link_node(&cl->pq_node, parent, p); | 312 | rb_link_node(&cl->pq_node, parent, p); |
307 | rb_insert_color(&cl->pq_node, &q->wait_pq[cl->level]); | 313 | rb_insert_color(&cl->pq_node, &q->hlevel[cl->level].wait_pq); |
308 | } | 314 | } |
309 | 315 | ||
310 | /** | 316 | /** |
@@ -331,7 +337,7 @@ static inline void htb_add_class_to_row(struct htb_sched *q, | |||
331 | while (mask) { | 337 | while (mask) { |
332 | int prio = ffz(~mask); | 338 | int prio = ffz(~mask); |
333 | mask &= ~(1 << prio); | 339 | mask &= ~(1 << prio); |
334 | htb_add_to_id_tree(q->row[cl->level] + prio, cl, prio); | 340 | htb_add_to_id_tree(&q->hlevel[cl->level].hprio[prio].row, cl, prio); |
335 | } | 341 | } |
336 | } | 342 | } |
337 | 343 | ||
@@ -357,16 +363,18 @@ static inline void htb_remove_class_from_row(struct htb_sched *q, | |||
357 | struct htb_class *cl, int mask) | 363 | struct htb_class *cl, int mask) |
358 | { | 364 | { |
359 | int m = 0; | 365 | int m = 0; |
366 | struct htb_level *hlevel = &q->hlevel[cl->level]; | ||
360 | 367 | ||
361 | while (mask) { | 368 | while (mask) { |
362 | int prio = ffz(~mask); | 369 | int prio = ffz(~mask); |
370 | struct htb_prio *hprio = &hlevel->hprio[prio]; | ||
363 | 371 | ||
364 | mask &= ~(1 << prio); | 372 | mask &= ~(1 << prio); |
365 | if (q->ptr[cl->level][prio] == cl->node + prio) | 373 | if (hprio->ptr == cl->node + prio) |
366 | htb_next_rb_node(q->ptr[cl->level] + prio); | 374 | htb_next_rb_node(&hprio->ptr); |
367 | 375 | ||
368 | htb_safe_rb_erase(cl->node + prio, q->row[cl->level] + prio); | 376 | htb_safe_rb_erase(cl->node + prio, &hprio->row); |
369 | if (!q->row[cl->level][prio].rb_node) | 377 | if (!hprio->row.rb_node) |
370 | m |= 1 << prio; | 378 | m |= 1 << prio; |
371 | } | 379 | } |
372 | q->row_mask[cl->level] &= ~m; | 380 | q->row_mask[cl->level] &= ~m; |
@@ -390,13 +398,13 @@ static void htb_activate_prios(struct htb_sched *q, struct htb_class *cl) | |||
390 | int prio = ffz(~m); | 398 | int prio = ffz(~m); |
391 | m &= ~(1 << prio); | 399 | m &= ~(1 << prio); |
392 | 400 | ||
393 | if (p->un.inner.feed[prio].rb_node) | 401 | if (p->un.inner.clprio[prio].feed.rb_node) |
394 | /* parent already has its feed in use so that | 402 | /* parent already has its feed in use so that |
395 | * reset bit in mask as parent is already ok | 403 | * reset bit in mask as parent is already ok |
396 | */ | 404 | */ |
397 | mask &= ~(1 << prio); | 405 | mask &= ~(1 << prio); |
398 | 406 | ||
399 | htb_add_to_id_tree(p->un.inner.feed + prio, cl, prio); | 407 | htb_add_to_id_tree(&p->un.inner.clprio[prio].feed, cl, prio); |
400 | } | 408 | } |
401 | p->prio_activity |= mask; | 409 | p->prio_activity |= mask; |
402 | cl = p; | 410 | cl = p; |
@@ -426,18 +434,19 @@ static void htb_deactivate_prios(struct htb_sched *q, struct htb_class *cl) | |||
426 | int prio = ffz(~m); | 434 | int prio = ffz(~m); |
427 | m &= ~(1 << prio); | 435 | m &= ~(1 << prio); |
428 | 436 | ||
429 | if (p->un.inner.ptr[prio] == cl->node + prio) { | 437 | if (p->un.inner.clprio[prio].ptr == cl->node + prio) { |
430 | /* we are removing child which is pointed to from | 438 | /* we are removing child which is pointed to from |
431 | * parent feed - forget the pointer but remember | 439 | * parent feed - forget the pointer but remember |
432 | * classid | 440 | * classid |
433 | */ | 441 | */ |
434 | p->un.inner.last_ptr_id[prio] = cl->common.classid; | 442 | p->un.inner.clprio[prio].last_ptr_id = cl->common.classid; |
435 | p->un.inner.ptr[prio] = NULL; | 443 | p->un.inner.clprio[prio].ptr = NULL; |
436 | } | 444 | } |
437 | 445 | ||
438 | htb_safe_rb_erase(cl->node + prio, p->un.inner.feed + prio); | 446 | htb_safe_rb_erase(cl->node + prio, |
447 | &p->un.inner.clprio[prio].feed); | ||
439 | 448 | ||
440 | if (!p->un.inner.feed[prio].rb_node) | 449 | if (!p->un.inner.clprio[prio].feed.rb_node) |
441 | mask |= 1 << prio; | 450 | mask |= 1 << prio; |
442 | } | 451 | } |
443 | 452 | ||
@@ -652,7 +661,7 @@ static void htb_charge_class(struct htb_sched *q, struct htb_class *cl, | |||
652 | htb_change_class_mode(q, cl, &diff); | 661 | htb_change_class_mode(q, cl, &diff); |
653 | if (old_mode != cl->cmode) { | 662 | if (old_mode != cl->cmode) { |
654 | if (old_mode != HTB_CAN_SEND) | 663 | if (old_mode != HTB_CAN_SEND) |
655 | htb_safe_rb_erase(&cl->pq_node, q->wait_pq + cl->level); | 664 | htb_safe_rb_erase(&cl->pq_node, &q->hlevel[cl->level].wait_pq); |
656 | if (cl->cmode != HTB_CAN_SEND) | 665 | if (cl->cmode != HTB_CAN_SEND) |
657 | htb_add_to_wait_tree(q, cl, diff); | 666 | htb_add_to_wait_tree(q, cl, diff); |
658 | } | 667 | } |
@@ -672,7 +681,7 @@ static void htb_charge_class(struct htb_sched *q, struct htb_class *cl, | |||
672 | * next pending event (0 for no event in pq, q->now for too many events). | 681 | * next pending event (0 for no event in pq, q->now for too many events). |
673 | * Note: Applied are events whose have cl->pq_key <= q->now. | 682 | * Note: Applied are events whose have cl->pq_key <= q->now. |
674 | */ | 683 | */ |
675 | static s64 htb_do_events(struct htb_sched *q, int level, | 684 | static s64 htb_do_events(struct htb_sched *q, const int level, |
676 | unsigned long start) | 685 | unsigned long start) |
677 | { | 686 | { |
678 | /* don't run for longer than 2 jiffies; 2 is used instead of | 687 | /* don't run for longer than 2 jiffies; 2 is used instead of |
@@ -680,10 +689,12 @@ static s64 htb_do_events(struct htb_sched *q, int level, | |||
680 | * too soon | 689 | * too soon |
681 | */ | 690 | */ |
682 | unsigned long stop_at = start + 2; | 691 | unsigned long stop_at = start + 2; |
692 | struct rb_root *wait_pq = &q->hlevel[level].wait_pq; | ||
693 | |||
683 | while (time_before(jiffies, stop_at)) { | 694 | while (time_before(jiffies, stop_at)) { |
684 | struct htb_class *cl; | 695 | struct htb_class *cl; |
685 | s64 diff; | 696 | s64 diff; |
686 | struct rb_node *p = rb_first(&q->wait_pq[level]); | 697 | struct rb_node *p = rb_first(wait_pq); |
687 | 698 | ||
688 | if (!p) | 699 | if (!p) |
689 | return 0; | 700 | return 0; |
@@ -692,7 +703,7 @@ static s64 htb_do_events(struct htb_sched *q, int level, | |||
692 | if (cl->pq_key > q->now) | 703 | if (cl->pq_key > q->now) |
693 | return cl->pq_key; | 704 | return cl->pq_key; |
694 | 705 | ||
695 | htb_safe_rb_erase(p, q->wait_pq + level); | 706 | htb_safe_rb_erase(p, wait_pq); |
696 | diff = min_t(s64, q->now - cl->t_c, cl->mbuffer); | 707 | diff = min_t(s64, q->now - cl->t_c, cl->mbuffer); |
697 | htb_change_class_mode(q, cl, &diff); | 708 | htb_change_class_mode(q, cl, &diff); |
698 | if (cl->cmode != HTB_CAN_SEND) | 709 | if (cl->cmode != HTB_CAN_SEND) |
@@ -736,8 +747,7 @@ static struct rb_node *htb_id_find_next_upper(int prio, struct rb_node *n, | |||
736 | * | 747 | * |
737 | * Find leaf where current feed pointers points to. | 748 | * Find leaf where current feed pointers points to. |
738 | */ | 749 | */ |
739 | static struct htb_class *htb_lookup_leaf(struct rb_root *tree, int prio, | 750 | static struct htb_class *htb_lookup_leaf(struct htb_prio *hprio, const int prio) |
740 | struct rb_node **pptr, u32 * pid) | ||
741 | { | 751 | { |
742 | int i; | 752 | int i; |
743 | struct { | 753 | struct { |
@@ -746,10 +756,10 @@ static struct htb_class *htb_lookup_leaf(struct rb_root *tree, int prio, | |||
746 | u32 *pid; | 756 | u32 *pid; |
747 | } stk[TC_HTB_MAXDEPTH], *sp = stk; | 757 | } stk[TC_HTB_MAXDEPTH], *sp = stk; |
748 | 758 | ||
749 | BUG_ON(!tree->rb_node); | 759 | BUG_ON(!hprio->row.rb_node); |
750 | sp->root = tree->rb_node; | 760 | sp->root = hprio->row.rb_node; |
751 | sp->pptr = pptr; | 761 | sp->pptr = &hprio->ptr; |
752 | sp->pid = pid; | 762 | sp->pid = &hprio->last_ptr_id; |
753 | 763 | ||
754 | for (i = 0; i < 65535; i++) { | 764 | for (i = 0; i < 65535; i++) { |
755 | if (!*sp->pptr && *sp->pid) { | 765 | if (!*sp->pptr && *sp->pid) { |
@@ -776,12 +786,15 @@ static struct htb_class *htb_lookup_leaf(struct rb_root *tree, int prio, | |||
776 | } | 786 | } |
777 | } else { | 787 | } else { |
778 | struct htb_class *cl; | 788 | struct htb_class *cl; |
789 | struct htb_prio *clp; | ||
790 | |||
779 | cl = rb_entry(*sp->pptr, struct htb_class, node[prio]); | 791 | cl = rb_entry(*sp->pptr, struct htb_class, node[prio]); |
780 | if (!cl->level) | 792 | if (!cl->level) |
781 | return cl; | 793 | return cl; |
782 | (++sp)->root = cl->un.inner.feed[prio].rb_node; | 794 | clp = &cl->un.inner.clprio[prio]; |
783 | sp->pptr = cl->un.inner.ptr + prio; | 795 | (++sp)->root = clp->feed.rb_node; |
784 | sp->pid = cl->un.inner.last_ptr_id + prio; | 796 | sp->pptr = &clp->ptr; |
797 | sp->pid = &clp->last_ptr_id; | ||
785 | } | 798 | } |
786 | } | 799 | } |
787 | WARN_ON(1); | 800 | WARN_ON(1); |
@@ -791,15 +804,16 @@ static struct htb_class *htb_lookup_leaf(struct rb_root *tree, int prio, | |||
791 | /* dequeues packet at given priority and level; call only if | 804 | /* dequeues packet at given priority and level; call only if |
792 | * you are sure that there is active class at prio/level | 805 | * you are sure that there is active class at prio/level |
793 | */ | 806 | */ |
794 | static struct sk_buff *htb_dequeue_tree(struct htb_sched *q, int prio, | 807 | static struct sk_buff *htb_dequeue_tree(struct htb_sched *q, const int prio, |
795 | int level) | 808 | const int level) |
796 | { | 809 | { |
797 | struct sk_buff *skb = NULL; | 810 | struct sk_buff *skb = NULL; |
798 | struct htb_class *cl, *start; | 811 | struct htb_class *cl, *start; |
812 | struct htb_level *hlevel = &q->hlevel[level]; | ||
813 | struct htb_prio *hprio = &hlevel->hprio[prio]; | ||
814 | |||
799 | /* look initial class up in the row */ | 815 | /* look initial class up in the row */ |
800 | start = cl = htb_lookup_leaf(q->row[level] + prio, prio, | 816 | start = cl = htb_lookup_leaf(hprio, prio); |
801 | q->ptr[level] + prio, | ||
802 | q->last_ptr_id[level] + prio); | ||
803 | 817 | ||
804 | do { | 818 | do { |
805 | next: | 819 | next: |
@@ -819,9 +833,7 @@ next: | |||
819 | if ((q->row_mask[level] & (1 << prio)) == 0) | 833 | if ((q->row_mask[level] & (1 << prio)) == 0) |
820 | return NULL; | 834 | return NULL; |
821 | 835 | ||
822 | next = htb_lookup_leaf(q->row[level] + prio, | 836 | next = htb_lookup_leaf(hprio, prio); |
823 | prio, q->ptr[level] + prio, | ||
824 | q->last_ptr_id[level] + prio); | ||
825 | 837 | ||
826 | if (cl == start) /* fix start if we just deleted it */ | 838 | if (cl == start) /* fix start if we just deleted it */ |
827 | start = next; | 839 | start = next; |
@@ -834,11 +846,9 @@ next: | |||
834 | break; | 846 | break; |
835 | 847 | ||
836 | qdisc_warn_nonwc("htb", cl->un.leaf.q); | 848 | qdisc_warn_nonwc("htb", cl->un.leaf.q); |
837 | htb_next_rb_node((level ? cl->parent->un.inner.ptr : q-> | 849 | htb_next_rb_node(level ? &cl->parent->un.inner.clprio[prio].ptr: |
838 | ptr[0]) + prio); | 850 | &q->hlevel[0].hprio[prio].ptr); |
839 | cl = htb_lookup_leaf(q->row[level] + prio, prio, | 851 | cl = htb_lookup_leaf(hprio, prio); |
840 | q->ptr[level] + prio, | ||
841 | q->last_ptr_id[level] + prio); | ||
842 | 852 | ||
843 | } while (cl != start); | 853 | } while (cl != start); |
844 | 854 | ||
@@ -847,8 +857,8 @@ next: | |||
847 | cl->un.leaf.deficit[level] -= qdisc_pkt_len(skb); | 857 | cl->un.leaf.deficit[level] -= qdisc_pkt_len(skb); |
848 | if (cl->un.leaf.deficit[level] < 0) { | 858 | if (cl->un.leaf.deficit[level] < 0) { |
849 | cl->un.leaf.deficit[level] += cl->quantum; | 859 | cl->un.leaf.deficit[level] += cl->quantum; |
850 | htb_next_rb_node((level ? cl->parent->un.inner.ptr : q-> | 860 | htb_next_rb_node(level ? &cl->parent->un.inner.clprio[prio].ptr : |
851 | ptr[0]) + prio); | 861 | &q->hlevel[0].hprio[prio].ptr); |
852 | } | 862 | } |
853 | /* this used to be after charge_class but this constelation | 863 | /* this used to be after charge_class but this constelation |
854 | * gives us slightly better performance | 864 | * gives us slightly better performance |
@@ -888,15 +898,14 @@ ok: | |||
888 | for (level = 0; level < TC_HTB_MAXDEPTH; level++) { | 898 | for (level = 0; level < TC_HTB_MAXDEPTH; level++) { |
889 | /* common case optimization - skip event handler quickly */ | 899 | /* common case optimization - skip event handler quickly */ |
890 | int m; | 900 | int m; |
891 | s64 event; | 901 | s64 event = q->near_ev_cache[level]; |
892 | 902 | ||
893 | if (q->now >= q->near_ev_cache[level]) { | 903 | if (q->now >= event) { |
894 | event = htb_do_events(q, level, start_at); | 904 | event = htb_do_events(q, level, start_at); |
895 | if (!event) | 905 | if (!event) |
896 | event = q->now + NSEC_PER_SEC; | 906 | event = q->now + NSEC_PER_SEC; |
897 | q->near_ev_cache[level] = event; | 907 | q->near_ev_cache[level] = event; |
898 | } else | 908 | } |
899 | event = q->near_ev_cache[level]; | ||
900 | 909 | ||
901 | if (next_event > event) | 910 | if (next_event > event) |
902 | next_event = event; | 911 | next_event = event; |
@@ -976,10 +985,8 @@ static void htb_reset(struct Qdisc *sch) | |||
976 | qdisc_watchdog_cancel(&q->watchdog); | 985 | qdisc_watchdog_cancel(&q->watchdog); |
977 | __skb_queue_purge(&q->direct_queue); | 986 | __skb_queue_purge(&q->direct_queue); |
978 | sch->q.qlen = 0; | 987 | sch->q.qlen = 0; |
979 | memset(q->row, 0, sizeof(q->row)); | 988 | memset(q->hlevel, 0, sizeof(q->hlevel)); |
980 | memset(q->row_mask, 0, sizeof(q->row_mask)); | 989 | memset(q->row_mask, 0, sizeof(q->row_mask)); |
981 | memset(q->wait_pq, 0, sizeof(q->wait_pq)); | ||
982 | memset(q->ptr, 0, sizeof(q->ptr)); | ||
983 | for (i = 0; i < TC_HTB_NUMPRIO; i++) | 990 | for (i = 0; i < TC_HTB_NUMPRIO; i++) |
984 | INIT_LIST_HEAD(q->drops + i); | 991 | INIT_LIST_HEAD(q->drops + i); |
985 | } | 992 | } |
@@ -1200,7 +1207,8 @@ static void htb_parent_to_leaf(struct htb_sched *q, struct htb_class *cl, | |||
1200 | WARN_ON(cl->level || !cl->un.leaf.q || cl->prio_activity); | 1207 | WARN_ON(cl->level || !cl->un.leaf.q || cl->prio_activity); |
1201 | 1208 | ||
1202 | if (parent->cmode != HTB_CAN_SEND) | 1209 | if (parent->cmode != HTB_CAN_SEND) |
1203 | htb_safe_rb_erase(&parent->pq_node, q->wait_pq + parent->level); | 1210 | htb_safe_rb_erase(&parent->pq_node, |
1211 | &q->hlevel[parent->level].wait_pq); | ||
1204 | 1212 | ||
1205 | parent->level = 0; | 1213 | parent->level = 0; |
1206 | memset(&parent->un.inner, 0, sizeof(parent->un.inner)); | 1214 | memset(&parent->un.inner, 0, sizeof(parent->un.inner)); |
@@ -1289,7 +1297,8 @@ static int htb_delete(struct Qdisc *sch, unsigned long arg) | |||
1289 | htb_deactivate(q, cl); | 1297 | htb_deactivate(q, cl); |
1290 | 1298 | ||
1291 | if (cl->cmode != HTB_CAN_SEND) | 1299 | if (cl->cmode != HTB_CAN_SEND) |
1292 | htb_safe_rb_erase(&cl->pq_node, q->wait_pq + cl->level); | 1300 | htb_safe_rb_erase(&cl->pq_node, |
1301 | &q->hlevel[cl->level].wait_pq); | ||
1293 | 1302 | ||
1294 | if (last_child) | 1303 | if (last_child) |
1295 | htb_parent_to_leaf(q, cl, new_q); | 1304 | htb_parent_to_leaf(q, cl, new_q); |
@@ -1411,7 +1420,7 @@ static int htb_change_class(struct Qdisc *sch, u32 classid, | |||
1411 | 1420 | ||
1412 | /* remove from evt list because of level change */ | 1421 | /* remove from evt list because of level change */ |
1413 | if (parent->cmode != HTB_CAN_SEND) { | 1422 | if (parent->cmode != HTB_CAN_SEND) { |
1414 | htb_safe_rb_erase(&parent->pq_node, q->wait_pq); | 1423 | htb_safe_rb_erase(&parent->pq_node, &q->hlevel[0].wait_pq); |
1415 | parent->cmode = HTB_CAN_SEND; | 1424 | parent->cmode = HTB_CAN_SEND; |
1416 | } | 1425 | } |
1417 | parent->level = (parent->parent ? parent->parent->level | 1426 | parent->level = (parent->parent ? parent->parent->level |