blob: 3e86fd3a1b783338622c140a62d6fb379659b209 [file] [log] [blame]
Stephen Hemminger87990462006-08-10 23:35:16 -07001/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07002 * net/sched/sch_htb.c Hierarchical token bucket, feed tree version
3 *
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version
7 * 2 of the License, or (at your option) any later version.
8 *
9 * Authors: Martin Devera, <devik@cdi.cz>
10 *
11 * Credits (in time order) for older HTB versions:
12 * Stef Coene <stef.coene@docum.org>
13 * HTB support at LARTC mailing list
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +090014 * Ondrej Kraus, <krauso@barr.cz>
Linus Torvalds1da177e2005-04-16 15:20:36 -070015 * found missing INIT_QDISC(htb)
16 * Vladimir Smelhaus, Aamer Akhter, Bert Hubert
17 * helped a lot to locate nasty class stall bug
18 * Andi Kleen, Jamal Hadi, Bert Hubert
19 * code review and helpful comments on shaping
20 * Tomasz Wrona, <tw@eter.tym.pl>
21 * created test case so that I was able to fix nasty bug
22 * Wilfried Weissmann
23 * spotted bug in dequeue code and helped with fix
24 * Jiri Fojtasek
25 * fixed requeue routine
26 * and many others. thanks.
Linus Torvalds1da177e2005-04-16 15:20:36 -070027 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070028#include <linux/module.h>
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -070029#include <linux/moduleparam.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070030#include <linux/types.h>
31#include <linux/kernel.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070032#include <linux/string.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070033#include <linux/errno.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070034#include <linux/skbuff.h>
35#include <linux/list.h>
36#include <linux/compiler.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070037#include <linux/rbtree.h>
Jarek Poplawski12247362009-02-01 01:13:22 -080038#include <linux/workqueue.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090039#include <linux/slab.h>
Patrick McHardy0ba48052007-07-02 22:49:07 -070040#include <net/netlink.h>
41#include <net/pkt_sched.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070042
43/* HTB algorithm.
44 Author: devik@cdi.cz
45 ========================================================================
46 HTB is like TBF with multiple classes. It is also similar to CBQ because
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +090047 it allows to assign priority to each class in hierarchy.
Linus Torvalds1da177e2005-04-16 15:20:36 -070048 In fact it is another implementation of Floyd's formal sharing.
49
50 Levels:
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +090051 Each class is assigned level. Leaf has ALWAYS level 0 and root
Linus Torvalds1da177e2005-04-16 15:20:36 -070052 classes have level TC_HTB_MAXDEPTH-1. Interior nodes has level
53 one less than their parent.
54*/
55
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -070056static int htb_hysteresis __read_mostly = 0; /* whether to use mode hysteresis for speedup */
Stephen Hemminger87990462006-08-10 23:35:16 -070057#define HTB_VER 0x30011 /* major must be matched with number suplied by TC as version */
Linus Torvalds1da177e2005-04-16 15:20:36 -070058
59#if HTB_VER >> 16 != TC_HTB_PROTOVER
60#error "Mismatched sch_htb.c and pkt_sch.h"
61#endif
62
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -070063/* Module parameter and sysfs export */
64module_param (htb_hysteresis, int, 0640);
65MODULE_PARM_DESC(htb_hysteresis, "Hysteresis mode, less CPU load, less accurate");
66
Linus Torvalds1da177e2005-04-16 15:20:36 -070067/* used internaly to keep status of single class */
68enum htb_cmode {
Stephen Hemminger87990462006-08-10 23:35:16 -070069 HTB_CANT_SEND, /* class can't send and can't borrow */
70 HTB_MAY_BORROW, /* class can't send but may borrow */
71 HTB_CAN_SEND /* class can send */
Linus Torvalds1da177e2005-04-16 15:20:36 -070072};
73
74/* interior & leaf nodes; props specific to leaves are marked L: */
Stephen Hemminger87990462006-08-10 23:35:16 -070075struct htb_class {
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -070076 struct Qdisc_class_common common;
Stephen Hemminger87990462006-08-10 23:35:16 -070077 /* general class parameters */
Eric Dumazetc1a8f1f2009-08-16 09:36:49 +000078 struct gnet_stats_basic_packed bstats;
Stephen Hemminger87990462006-08-10 23:35:16 -070079 struct gnet_stats_queue qstats;
80 struct gnet_stats_rate_est rate_est;
81 struct tc_htb_xstats xstats; /* our special stats */
82 int refcnt; /* usage count of this class */
Linus Torvalds1da177e2005-04-16 15:20:36 -070083
Stephen Hemminger87990462006-08-10 23:35:16 -070084 /* topology */
85 int level; /* our level (see above) */
Patrick McHardy42077592008-07-05 23:22:53 -070086 unsigned int children;
Stephen Hemminger87990462006-08-10 23:35:16 -070087 struct htb_class *parent; /* parent class */
Linus Torvalds1da177e2005-04-16 15:20:36 -070088
Jarek Poplawskic19f7a32008-12-03 21:09:45 -080089 int prio; /* these two are used only by leaves... */
90 int quantum; /* but stored for parent-to-leaf return */
91
Stephen Hemminger87990462006-08-10 23:35:16 -070092 union {
93 struct htb_class_leaf {
94 struct Qdisc *q;
Stephen Hemminger87990462006-08-10 23:35:16 -070095 int deficit[TC_HTB_MAXDEPTH];
96 struct list_head drop_list;
97 } leaf;
98 struct htb_class_inner {
99 struct rb_root feed[TC_HTB_NUMPRIO]; /* feed trees */
100 struct rb_node *ptr[TC_HTB_NUMPRIO]; /* current class ptr */
101 /* When class changes from state 1->2 and disconnects from
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000102 * parent's feed then we lost ptr value and start from the
103 * first child again. Here we store classid of the
104 * last valid ptr (used when ptr is NULL).
105 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700106 u32 last_ptr_id[TC_HTB_NUMPRIO];
107 } inner;
108 } un;
109 struct rb_node node[TC_HTB_NUMPRIO]; /* node for self or feed tree */
110 struct rb_node pq_node; /* node for event queue */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700111 psched_time_t pq_key;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700112
Stephen Hemminger87990462006-08-10 23:35:16 -0700113 int prio_activity; /* for which prios are we active */
114 enum htb_cmode cmode; /* current mode of the class */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700115
Stephen Hemminger87990462006-08-10 23:35:16 -0700116 /* class attached filters */
117 struct tcf_proto *filter_list;
118 int filter_cnt;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700119
Stephen Hemminger87990462006-08-10 23:35:16 -0700120 /* token bucket parameters */
121 struct qdisc_rate_table *rate; /* rate table of the class itself */
122 struct qdisc_rate_table *ceil; /* ceiling rate (limits borrows too) */
123 long buffer, cbuffer; /* token bucket depth/rate */
124 psched_tdiff_t mbuffer; /* max wait time */
125 long tokens, ctokens; /* current number of tokens */
126 psched_time_t t_c; /* checkpoint time */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700127};
128
Stephen Hemminger87990462006-08-10 23:35:16 -0700129struct htb_sched {
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700130 struct Qdisc_class_hash clhash;
Stephen Hemminger0cef2962006-08-10 23:35:38 -0700131 struct list_head drops[TC_HTB_NUMPRIO];/* active leaves (for drops) */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700132
Stephen Hemminger87990462006-08-10 23:35:16 -0700133 /* self list - roots of self generating tree */
134 struct rb_root row[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO];
135 int row_mask[TC_HTB_MAXDEPTH];
136 struct rb_node *ptr[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO];
137 u32 last_ptr_id[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700138
Stephen Hemminger87990462006-08-10 23:35:16 -0700139 /* self wait list - roots of wait PQs per row */
140 struct rb_root wait_pq[TC_HTB_MAXDEPTH];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700141
Stephen Hemminger87990462006-08-10 23:35:16 -0700142 /* time of nearest event per level (row) */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700143 psched_time_t near_ev_cache[TC_HTB_MAXDEPTH];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700144
Stephen Hemminger87990462006-08-10 23:35:16 -0700145 int defcls; /* class where unclassified flows go to */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700146
Stephen Hemminger87990462006-08-10 23:35:16 -0700147 /* filters for qdisc itself */
148 struct tcf_proto *filter_list;
Stephen Hemminger87990462006-08-10 23:35:16 -0700149
150 int rate2quantum; /* quant = rate / rate2quantum */
151 psched_time_t now; /* cached dequeue time */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700152 struct qdisc_watchdog watchdog;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700153
Stephen Hemminger87990462006-08-10 23:35:16 -0700154 /* non shaped skbs; let them go directly thru */
155 struct sk_buff_head direct_queue;
156 int direct_qlen; /* max qlen of above */
157
158 long direct_pkts;
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800159
160#define HTB_WARN_TOOMANYEVENTS 0x1
161 unsigned int warned; /* only one warning */
Jarek Poplawski12247362009-02-01 01:13:22 -0800162 struct work_struct work;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700163};
164
Linus Torvalds1da177e2005-04-16 15:20:36 -0700165/* find class in global hash table using given handle */
Stephen Hemminger87990462006-08-10 23:35:16 -0700166static inline struct htb_class *htb_find(u32 handle, struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700167{
168 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700169 struct Qdisc_class_common *clc;
Stephen Hemminger0cef2962006-08-10 23:35:38 -0700170
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700171 clc = qdisc_class_find(&q->clhash, handle);
172 if (clc == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700173 return NULL;
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700174 return container_of(clc, struct htb_class, common);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700175}
176
177/**
178 * htb_classify - classify a packet into class
179 *
180 * It returns NULL if the packet should be dropped or -1 if the packet
181 * should be passed directly thru. In all other cases leaf class is returned.
182 * We allow direct class selection by classid in priority. The we examine
183 * filters in qdisc and in inner nodes (if higher filter points to the inner
184 * node). If we end up with classid MAJOR:0 we enqueue the skb into special
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900185 * internal fifo (direct). These packets then go directly thru. If we still
Linus Torvalds1da177e2005-04-16 15:20:36 -0700186 * have no valid leaf we try to use MAJOR:default leaf. It still unsuccessfull
187 * then finish and return direct queue.
188 */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000189#define HTB_DIRECT ((struct htb_class *)-1L)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700190
Stephen Hemminger87990462006-08-10 23:35:16 -0700191static struct htb_class *htb_classify(struct sk_buff *skb, struct Qdisc *sch,
192 int *qerr)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700193{
194 struct htb_sched *q = qdisc_priv(sch);
195 struct htb_class *cl;
196 struct tcf_result res;
197 struct tcf_proto *tcf;
198 int result;
199
200 /* allow to select class by setting skb->priority to valid classid;
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000201 * note that nfmark can be used too by attaching filter fw with no
202 * rules in it
203 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700204 if (skb->priority == sch->handle)
Stephen Hemminger87990462006-08-10 23:35:16 -0700205 return HTB_DIRECT; /* X:0 (direct flow) selected */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000206 cl = htb_find(skb->priority, sch);
207 if (cl && cl->level == 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700208 return cl;
209
Jarek Poplawskic27f3392008-08-04 22:39:11 -0700210 *qerr = NET_XMIT_SUCCESS | __NET_XMIT_BYPASS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700211 tcf = q->filter_list;
212 while (tcf && (result = tc_classify(skb, tcf, &res)) >= 0) {
213#ifdef CONFIG_NET_CLS_ACT
214 switch (result) {
215 case TC_ACT_QUEUED:
Stephen Hemminger87990462006-08-10 23:35:16 -0700216 case TC_ACT_STOLEN:
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700217 *qerr = NET_XMIT_SUCCESS | __NET_XMIT_STOLEN;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700218 case TC_ACT_SHOT:
219 return NULL;
220 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700221#endif
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000222 cl = (void *)res.class;
223 if (!cl) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700224 if (res.classid == sch->handle)
Stephen Hemminger87990462006-08-10 23:35:16 -0700225 return HTB_DIRECT; /* X:0 (direct flow) */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000226 cl = htb_find(res.classid, sch);
227 if (!cl)
Stephen Hemminger87990462006-08-10 23:35:16 -0700228 break; /* filter selected invalid classid */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700229 }
230 if (!cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -0700231 return cl; /* we hit leaf; return it */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700232
233 /* we have got inner class; apply inner filter chain */
234 tcf = cl->filter_list;
235 }
236 /* classification failed; try to use default class */
Stephen Hemminger87990462006-08-10 23:35:16 -0700237 cl = htb_find(TC_H_MAKE(TC_H_MAJ(sch->handle), q->defcls), sch);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700238 if (!cl || cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -0700239 return HTB_DIRECT; /* bad default .. this is safe bet */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700240 return cl;
241}
242
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243/**
244 * htb_add_to_id_tree - adds class to the round robin list
245 *
246 * Routine adds class to the list (actually tree) sorted by classid.
247 * Make sure that class is not already on such list for given prio.
248 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700249static void htb_add_to_id_tree(struct rb_root *root,
250 struct htb_class *cl, int prio)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700251{
252 struct rb_node **p = &root->rb_node, *parent = NULL;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700253
Linus Torvalds1da177e2005-04-16 15:20:36 -0700254 while (*p) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700255 struct htb_class *c;
256 parent = *p;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257 c = rb_entry(parent, struct htb_class, node[prio]);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700258
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700259 if (cl->common.classid > c->common.classid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700260 p = &parent->rb_right;
Stephen Hemminger87990462006-08-10 23:35:16 -0700261 else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700262 p = &parent->rb_left;
263 }
264 rb_link_node(&cl->node[prio], parent, p);
265 rb_insert_color(&cl->node[prio], root);
266}
267
268/**
269 * htb_add_to_wait_tree - adds class to the event queue with delay
270 *
271 * The class is added to priority event queue to indicate that class will
272 * change its mode in cl->pq_key microseconds. Make sure that class is not
273 * already in the queue.
274 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700275static void htb_add_to_wait_tree(struct htb_sched *q,
276 struct htb_class *cl, long delay)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700277{
278 struct rb_node **p = &q->wait_pq[cl->level].rb_node, *parent = NULL;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700279
Patrick McHardyfb983d42007-03-16 01:22:39 -0700280 cl->pq_key = q->now + delay;
281 if (cl->pq_key == q->now)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282 cl->pq_key++;
283
284 /* update the nearest event cache */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700285 if (q->near_ev_cache[cl->level] > cl->pq_key)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700286 q->near_ev_cache[cl->level] = cl->pq_key;
Stephen Hemminger87990462006-08-10 23:35:16 -0700287
Linus Torvalds1da177e2005-04-16 15:20:36 -0700288 while (*p) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700289 struct htb_class *c;
290 parent = *p;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700291 c = rb_entry(parent, struct htb_class, pq_node);
Patrick McHardyfb983d42007-03-16 01:22:39 -0700292 if (cl->pq_key >= c->pq_key)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700293 p = &parent->rb_right;
Stephen Hemminger87990462006-08-10 23:35:16 -0700294 else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700295 p = &parent->rb_left;
296 }
297 rb_link_node(&cl->pq_node, parent, p);
298 rb_insert_color(&cl->pq_node, &q->wait_pq[cl->level]);
299}
300
301/**
302 * htb_next_rb_node - finds next node in binary tree
303 *
304 * When we are past last key we return NULL.
305 * Average complexity is 2 steps per call.
306 */
Stephen Hemminger3696f622006-08-10 23:36:01 -0700307static inline void htb_next_rb_node(struct rb_node **n)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700308{
309 *n = rb_next(*n);
310}
311
312/**
313 * htb_add_class_to_row - add class to its row
314 *
315 * The class is added to row at priorities marked in mask.
316 * It does nothing if mask == 0.
317 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700318static inline void htb_add_class_to_row(struct htb_sched *q,
319 struct htb_class *cl, int mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700320{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700321 q->row_mask[cl->level] |= mask;
322 while (mask) {
323 int prio = ffz(~mask);
324 mask &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700325 htb_add_to_id_tree(q->row[cl->level] + prio, cl, prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700326 }
327}
328
Stephen Hemminger3696f622006-08-10 23:36:01 -0700329/* If this triggers, it is a bug in this code, but it need not be fatal */
330static void htb_safe_rb_erase(struct rb_node *rb, struct rb_root *root)
331{
Ismail Donmez81771b32006-10-03 13:49:10 -0700332 if (RB_EMPTY_NODE(rb)) {
Stephen Hemminger3696f622006-08-10 23:36:01 -0700333 WARN_ON(1);
334 } else {
335 rb_erase(rb, root);
336 RB_CLEAR_NODE(rb);
337 }
338}
339
340
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341/**
342 * htb_remove_class_from_row - removes class from its row
343 *
344 * The class is removed from row at priorities marked in mask.
345 * It does nothing if mask == 0.
346 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700347static inline void htb_remove_class_from_row(struct htb_sched *q,
348 struct htb_class *cl, int mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700349{
350 int m = 0;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700351
Linus Torvalds1da177e2005-04-16 15:20:36 -0700352 while (mask) {
353 int prio = ffz(~mask);
Stephen Hemminger3696f622006-08-10 23:36:01 -0700354
Linus Torvalds1da177e2005-04-16 15:20:36 -0700355 mask &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700356 if (q->ptr[cl->level][prio] == cl->node + prio)
357 htb_next_rb_node(q->ptr[cl->level] + prio);
Stephen Hemminger3696f622006-08-10 23:36:01 -0700358
359 htb_safe_rb_erase(cl->node + prio, q->row[cl->level] + prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700360 if (!q->row[cl->level][prio].rb_node)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700361 m |= 1 << prio;
362 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700363 q->row_mask[cl->level] &= ~m;
364}
365
366/**
367 * htb_activate_prios - creates active classe's feed chain
368 *
369 * The class is connected to ancestors and/or appropriate rows
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900370 * for priorities it is participating on. cl->cmode must be new
Linus Torvalds1da177e2005-04-16 15:20:36 -0700371 * (activated) mode. It does nothing if cl->prio_activity == 0.
372 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700373static void htb_activate_prios(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700374{
375 struct htb_class *p = cl->parent;
Stephen Hemminger87990462006-08-10 23:35:16 -0700376 long m, mask = cl->prio_activity;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700377
378 while (cl->cmode == HTB_MAY_BORROW && p && mask) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700379 m = mask;
380 while (m) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700381 int prio = ffz(~m);
382 m &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700383
Linus Torvalds1da177e2005-04-16 15:20:36 -0700384 if (p->un.inner.feed[prio].rb_node)
385 /* parent already has its feed in use so that
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000386 * reset bit in mask as parent is already ok
387 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700388 mask &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700389
390 htb_add_to_id_tree(p->un.inner.feed + prio, cl, prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700391 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700392 p->prio_activity |= mask;
Stephen Hemminger87990462006-08-10 23:35:16 -0700393 cl = p;
394 p = cl->parent;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700395
Linus Torvalds1da177e2005-04-16 15:20:36 -0700396 }
397 if (cl->cmode == HTB_CAN_SEND && mask)
Stephen Hemminger87990462006-08-10 23:35:16 -0700398 htb_add_class_to_row(q, cl, mask);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700399}
400
401/**
402 * htb_deactivate_prios - remove class from feed chain
403 *
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900404 * cl->cmode must represent old mode (before deactivation). It does
Linus Torvalds1da177e2005-04-16 15:20:36 -0700405 * nothing if cl->prio_activity == 0. Class is removed from all feed
406 * chains and rows.
407 */
408static void htb_deactivate_prios(struct htb_sched *q, struct htb_class *cl)
409{
410 struct htb_class *p = cl->parent;
Stephen Hemminger87990462006-08-10 23:35:16 -0700411 long m, mask = cl->prio_activity;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700412
413 while (cl->cmode == HTB_MAY_BORROW && p && mask) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700414 m = mask;
415 mask = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416 while (m) {
417 int prio = ffz(~m);
418 m &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700419
420 if (p->un.inner.ptr[prio] == cl->node + prio) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700421 /* we are removing child which is pointed to from
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000422 * parent feed - forget the pointer but remember
423 * classid
424 */
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700425 p->un.inner.last_ptr_id[prio] = cl->common.classid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700426 p->un.inner.ptr[prio] = NULL;
427 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700428
Stephen Hemminger3696f622006-08-10 23:36:01 -0700429 htb_safe_rb_erase(cl->node + prio, p->un.inner.feed + prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700430
431 if (!p->un.inner.feed[prio].rb_node)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700432 mask |= 1 << prio;
433 }
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700434
Linus Torvalds1da177e2005-04-16 15:20:36 -0700435 p->prio_activity &= ~mask;
Stephen Hemminger87990462006-08-10 23:35:16 -0700436 cl = p;
437 p = cl->parent;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700438
Linus Torvalds1da177e2005-04-16 15:20:36 -0700439 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700440 if (cl->cmode == HTB_CAN_SEND && mask)
441 htb_remove_class_from_row(q, cl, mask);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700442}
443
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700444static inline long htb_lowater(const struct htb_class *cl)
445{
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700446 if (htb_hysteresis)
447 return cl->cmode != HTB_CANT_SEND ? -cl->cbuffer : 0;
448 else
449 return 0;
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700450}
451static inline long htb_hiwater(const struct htb_class *cl)
452{
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700453 if (htb_hysteresis)
454 return cl->cmode == HTB_CAN_SEND ? -cl->buffer : 0;
455 else
456 return 0;
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700457}
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700458
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700459
Linus Torvalds1da177e2005-04-16 15:20:36 -0700460/**
461 * htb_class_mode - computes and returns current class mode
462 *
463 * It computes cl's mode at time cl->t_c+diff and returns it. If mode
464 * is not HTB_CAN_SEND then cl->pq_key is updated to time difference
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900465 * from now to time when cl will change its state.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700466 * Also it is worth to note that class mode doesn't change simply
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900467 * at cl->{c,}tokens == 0 but there can rather be hysteresis of
Linus Torvalds1da177e2005-04-16 15:20:36 -0700468 * 0 .. -cl->{c,}buffer range. It is meant to limit number of
469 * mode transitions per time unit. The speed gain is about 1/6.
470 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700471static inline enum htb_cmode
472htb_class_mode(struct htb_class *cl, long *diff)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700473{
Stephen Hemminger87990462006-08-10 23:35:16 -0700474 long toks;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700475
Stephen Hemminger87990462006-08-10 23:35:16 -0700476 if ((toks = (cl->ctokens + *diff)) < htb_lowater(cl)) {
477 *diff = -toks;
478 return HTB_CANT_SEND;
479 }
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700480
Stephen Hemminger87990462006-08-10 23:35:16 -0700481 if ((toks = (cl->tokens + *diff)) >= htb_hiwater(cl))
482 return HTB_CAN_SEND;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700483
Stephen Hemminger87990462006-08-10 23:35:16 -0700484 *diff = -toks;
485 return HTB_MAY_BORROW;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700486}
487
488/**
489 * htb_change_class_mode - changes classe's mode
490 *
491 * This should be the only way how to change classe's mode under normal
492 * cirsumstances. Routine will update feed lists linkage, change mode
493 * and add class to the wait event queue if appropriate. New mode should
494 * be different from old one and cl->pq_key has to be valid if changing
495 * to mode other than HTB_CAN_SEND (see htb_add_to_wait_tree).
496 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700497static void
Linus Torvalds1da177e2005-04-16 15:20:36 -0700498htb_change_class_mode(struct htb_sched *q, struct htb_class *cl, long *diff)
Stephen Hemminger87990462006-08-10 23:35:16 -0700499{
500 enum htb_cmode new_mode = htb_class_mode(cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700501
502 if (new_mode == cl->cmode)
Stephen Hemminger87990462006-08-10 23:35:16 -0700503 return;
504
505 if (cl->prio_activity) { /* not necessary: speed optimization */
506 if (cl->cmode != HTB_CANT_SEND)
507 htb_deactivate_prios(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700508 cl->cmode = new_mode;
Stephen Hemminger87990462006-08-10 23:35:16 -0700509 if (new_mode != HTB_CANT_SEND)
510 htb_activate_prios(q, cl);
511 } else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700512 cl->cmode = new_mode;
513}
514
515/**
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900516 * htb_activate - inserts leaf cl into appropriate active feeds
Linus Torvalds1da177e2005-04-16 15:20:36 -0700517 *
518 * Routine learns (new) priority of leaf and activates feed chain
519 * for the prio. It can be called on already active leaf safely.
520 * It also adds leaf into droplist.
521 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700522static inline void htb_activate(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700523{
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700524 WARN_ON(cl->level || !cl->un.leaf.q || !cl->un.leaf.q->q.qlen);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700525
Linus Torvalds1da177e2005-04-16 15:20:36 -0700526 if (!cl->prio_activity) {
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800527 cl->prio_activity = 1 << cl->prio;
Stephen Hemminger87990462006-08-10 23:35:16 -0700528 htb_activate_prios(q, cl);
529 list_add_tail(&cl->un.leaf.drop_list,
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800530 q->drops + cl->prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700531 }
532}
533
534/**
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900535 * htb_deactivate - remove leaf cl from active feeds
Linus Torvalds1da177e2005-04-16 15:20:36 -0700536 *
537 * Make sure that leaf is active. In the other words it can't be called
538 * with non-active leaf. It also removes class from the drop list.
539 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700540static inline void htb_deactivate(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700541{
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700542 WARN_ON(!cl->prio_activity);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700543
Stephen Hemminger87990462006-08-10 23:35:16 -0700544 htb_deactivate_prios(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545 cl->prio_activity = 0;
546 list_del_init(&cl->un.leaf.drop_list);
547}
548
549static int htb_enqueue(struct sk_buff *skb, struct Qdisc *sch)
550{
Jarek Poplawskif30ab412008-11-13 22:56:30 -0800551 int uninitialized_var(ret);
Stephen Hemminger87990462006-08-10 23:35:16 -0700552 struct htb_sched *q = qdisc_priv(sch);
553 struct htb_class *cl = htb_classify(skb, sch, &ret);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700554
Stephen Hemminger87990462006-08-10 23:35:16 -0700555 if (cl == HTB_DIRECT) {
556 /* enqueue to helper queue */
557 if (q->direct_queue.qlen < q->direct_qlen) {
558 __skb_queue_tail(&q->direct_queue, skb);
559 q->direct_pkts++;
560 } else {
561 kfree_skb(skb);
562 sch->qstats.drops++;
563 return NET_XMIT_DROP;
564 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700565#ifdef CONFIG_NET_CLS_ACT
Stephen Hemminger87990462006-08-10 23:35:16 -0700566 } else if (!cl) {
Jarek Poplawskic27f3392008-08-04 22:39:11 -0700567 if (ret & __NET_XMIT_BYPASS)
Stephen Hemminger87990462006-08-10 23:35:16 -0700568 sch->qstats.drops++;
569 kfree_skb(skb);
570 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700571#endif
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700572 } else if ((ret = qdisc_enqueue(skb, cl->un.leaf.q)) != NET_XMIT_SUCCESS) {
573 if (net_xmit_drop_count(ret)) {
574 sch->qstats.drops++;
575 cl->qstats.drops++;
576 }
David S. Miller69747652008-08-17 23:55:36 -0700577 return ret;
Stephen Hemminger87990462006-08-10 23:35:16 -0700578 } else {
Eric Dumazetbfe0d022011-01-09 08:30:54 +0000579 bstats_update(&cl->bstats, skb);
Stephen Hemminger87990462006-08-10 23:35:16 -0700580 htb_activate(q, cl);
581 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700582
Stephen Hemminger87990462006-08-10 23:35:16 -0700583 sch->q.qlen++;
Eric Dumazetbfe0d022011-01-09 08:30:54 +0000584 qdisc_bstats_update(sch, skb);
Stephen Hemminger87990462006-08-10 23:35:16 -0700585 return NET_XMIT_SUCCESS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700586}
587
Jarek Poplawski59e42202008-12-03 21:17:27 -0800588static inline void htb_accnt_tokens(struct htb_class *cl, int bytes, long diff)
589{
590 long toks = diff + cl->tokens;
591
592 if (toks > cl->buffer)
593 toks = cl->buffer;
594 toks -= (long) qdisc_l2t(cl->rate, bytes);
595 if (toks <= -cl->mbuffer)
596 toks = 1 - cl->mbuffer;
597
598 cl->tokens = toks;
599}
600
601static inline void htb_accnt_ctokens(struct htb_class *cl, int bytes, long diff)
602{
603 long toks = diff + cl->ctokens;
604
605 if (toks > cl->cbuffer)
606 toks = cl->cbuffer;
607 toks -= (long) qdisc_l2t(cl->ceil, bytes);
608 if (toks <= -cl->mbuffer)
609 toks = 1 - cl->mbuffer;
610
611 cl->ctokens = toks;
612}
613
Linus Torvalds1da177e2005-04-16 15:20:36 -0700614/**
615 * htb_charge_class - charges amount "bytes" to leaf and ancestors
616 *
617 * Routine assumes that packet "bytes" long was dequeued from leaf cl
618 * borrowing from "level". It accounts bytes to ceil leaky bucket for
619 * leaf and all ancestors and to rate bucket for ancestors at levels
620 * "level" and higher. It also handles possible change of mode resulting
621 * from the update. Note that mode can also increase here (MAY_BORROW to
622 * CAN_SEND) because we can use more precise clock that event queue here.
623 * In such case we remove class from event queue first.
624 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700625static void htb_charge_class(struct htb_sched *q, struct htb_class *cl,
Ranjit Manomohanc9726d62007-07-10 22:43:16 -0700626 int level, struct sk_buff *skb)
Stephen Hemminger87990462006-08-10 23:35:16 -0700627{
Jussi Kivilinna0abf77e2008-07-20 00:08:27 -0700628 int bytes = qdisc_pkt_len(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700629 enum htb_cmode old_mode;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800630 long diff;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700631
632 while (cl) {
Patrick McHardy03cc45c2007-03-23 11:29:11 -0700633 diff = psched_tdiff_bounded(q->now, cl->t_c, cl->mbuffer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700634 if (cl->level >= level) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700635 if (cl->level == level)
636 cl->xstats.lends++;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800637 htb_accnt_tokens(cl, bytes, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700638 } else {
639 cl->xstats.borrows++;
Stephen Hemminger87990462006-08-10 23:35:16 -0700640 cl->tokens += diff; /* we moved t_c; update tokens */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700641 }
Jarek Poplawski59e42202008-12-03 21:17:27 -0800642 htb_accnt_ctokens(cl, bytes, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700643 cl->t_c = q->now;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700644
Stephen Hemminger87990462006-08-10 23:35:16 -0700645 old_mode = cl->cmode;
646 diff = 0;
647 htb_change_class_mode(q, cl, &diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700648 if (old_mode != cl->cmode) {
649 if (old_mode != HTB_CAN_SEND)
Stephen Hemminger3696f622006-08-10 23:36:01 -0700650 htb_safe_rb_erase(&cl->pq_node, q->wait_pq + cl->level);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700651 if (cl->cmode != HTB_CAN_SEND)
Stephen Hemminger87990462006-08-10 23:35:16 -0700652 htb_add_to_wait_tree(q, cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700653 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700654
Eric Dumazetbfe0d022011-01-09 08:30:54 +0000655 /* update basic stats except for leaves which are already updated */
656 if (cl->level)
657 bstats_update(&cl->bstats, skb);
658
Linus Torvalds1da177e2005-04-16 15:20:36 -0700659 cl = cl->parent;
660 }
661}
662
663/**
664 * htb_do_events - make mode changes to classes at the level
665 *
Patrick McHardyfb983d42007-03-16 01:22:39 -0700666 * Scans event queue for pending events and applies them. Returns time of
Jarek Poplawski12247362009-02-01 01:13:22 -0800667 * next pending event (0 for no event in pq, q->now for too many events).
Patrick McHardyfb983d42007-03-16 01:22:39 -0700668 * Note: Applied are events whose have cl->pq_key <= q->now.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700669 */
Jarek Poplawskia73be042009-01-12 21:54:40 -0800670static psched_time_t htb_do_events(struct htb_sched *q, int level,
671 unsigned long start)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700672{
Martin Devera8f3ea332008-03-23 22:00:38 -0700673 /* don't run for longer than 2 jiffies; 2 is used instead of
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000674 * 1 to simplify things when jiffy is going to be incremented
675 * too soon
676 */
Jarek Poplawskia73be042009-01-12 21:54:40 -0800677 unsigned long stop_at = start + 2;
Martin Devera8f3ea332008-03-23 22:00:38 -0700678 while (time_before(jiffies, stop_at)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700679 struct htb_class *cl;
680 long diff;
Akinbou Mita30bdbe32006-10-12 01:52:05 -0700681 struct rb_node *p = rb_first(&q->wait_pq[level]);
682
Stephen Hemminger87990462006-08-10 23:35:16 -0700683 if (!p)
684 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700685
686 cl = rb_entry(p, struct htb_class, pq_node);
Patrick McHardyfb983d42007-03-16 01:22:39 -0700687 if (cl->pq_key > q->now)
688 return cl->pq_key;
689
Stephen Hemminger3696f622006-08-10 23:36:01 -0700690 htb_safe_rb_erase(p, q->wait_pq + level);
Patrick McHardy03cc45c2007-03-23 11:29:11 -0700691 diff = psched_tdiff_bounded(q->now, cl->t_c, cl->mbuffer);
Stephen Hemminger87990462006-08-10 23:35:16 -0700692 htb_change_class_mode(q, cl, &diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700693 if (cl->cmode != HTB_CAN_SEND)
Stephen Hemminger87990462006-08-10 23:35:16 -0700694 htb_add_to_wait_tree(q, cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700695 }
Jarek Poplawski12247362009-02-01 01:13:22 -0800696
697 /* too much load - let's continue after a break for scheduling */
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800698 if (!(q->warned & HTB_WARN_TOOMANYEVENTS)) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000699 pr_warning("htb: too many events!\n");
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800700 q->warned |= HTB_WARN_TOOMANYEVENTS;
701 }
Jarek Poplawski12247362009-02-01 01:13:22 -0800702
703 return q->now;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700704}
705
706/* Returns class->node+prio from id-tree where classe's id is >= id. NULL
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000707 * is no such one exists.
708 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700709static struct rb_node *htb_id_find_next_upper(int prio, struct rb_node *n,
710 u32 id)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700711{
712 struct rb_node *r = NULL;
713 while (n) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700714 struct htb_class *cl =
715 rb_entry(n, struct htb_class, node[prio]);
Stephen Hemminger87990462006-08-10 23:35:16 -0700716
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700717 if (id > cl->common.classid) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700718 n = n->rb_right;
Jarek Poplawski1b5c0072008-12-09 22:34:40 -0800719 } else if (id < cl->common.classid) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700720 r = n;
721 n = n->rb_left;
Jarek Poplawski1b5c0072008-12-09 22:34:40 -0800722 } else {
723 return n;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700724 }
725 }
726 return r;
727}
728
729/**
730 * htb_lookup_leaf - returns next leaf class in DRR order
731 *
732 * Find leaf where current feed pointers points to.
733 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700734static struct htb_class *htb_lookup_leaf(struct rb_root *tree, int prio,
735 struct rb_node **pptr, u32 * pid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700736{
737 int i;
738 struct {
739 struct rb_node *root;
740 struct rb_node **pptr;
741 u32 *pid;
Stephen Hemminger87990462006-08-10 23:35:16 -0700742 } stk[TC_HTB_MAXDEPTH], *sp = stk;
743
Jarek Poplawski512bb432008-12-09 22:35:02 -0800744 BUG_ON(!tree->rb_node);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700745 sp->root = tree->rb_node;
746 sp->pptr = pptr;
747 sp->pid = pid;
748
749 for (i = 0; i < 65535; i++) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700750 if (!*sp->pptr && *sp->pid) {
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900751 /* ptr was invalidated but id is valid - try to recover
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000752 * the original or next ptr
753 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700754 *sp->pptr =
755 htb_id_find_next_upper(prio, sp->root, *sp->pid);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700757 *sp->pid = 0; /* ptr is valid now so that remove this hint as it
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000758 * can become out of date quickly
759 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700760 if (!*sp->pptr) { /* we are at right end; rewind & go up */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700761 *sp->pptr = sp->root;
Stephen Hemminger87990462006-08-10 23:35:16 -0700762 while ((*sp->pptr)->rb_left)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700763 *sp->pptr = (*sp->pptr)->rb_left;
764 if (sp > stk) {
765 sp--;
Jarek Poplawski512bb432008-12-09 22:35:02 -0800766 if (!*sp->pptr) {
767 WARN_ON(1);
Stephen Hemminger87990462006-08-10 23:35:16 -0700768 return NULL;
Jarek Poplawski512bb432008-12-09 22:35:02 -0800769 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700770 htb_next_rb_node(sp->pptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700771 }
772 } else {
773 struct htb_class *cl;
Stephen Hemminger87990462006-08-10 23:35:16 -0700774 cl = rb_entry(*sp->pptr, struct htb_class, node[prio]);
775 if (!cl->level)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700776 return cl;
777 (++sp)->root = cl->un.inner.feed[prio].rb_node;
Stephen Hemminger87990462006-08-10 23:35:16 -0700778 sp->pptr = cl->un.inner.ptr + prio;
779 sp->pid = cl->un.inner.last_ptr_id + prio;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700780 }
781 }
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700782 WARN_ON(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700783 return NULL;
784}
785
786/* dequeues packet at given priority and level; call only if
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000787 * you are sure that there is active class at prio/level
788 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700789static struct sk_buff *htb_dequeue_tree(struct htb_sched *q, int prio,
790 int level)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700791{
792 struct sk_buff *skb = NULL;
Stephen Hemminger87990462006-08-10 23:35:16 -0700793 struct htb_class *cl, *start;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700794 /* look initial class up in the row */
Stephen Hemminger87990462006-08-10 23:35:16 -0700795 start = cl = htb_lookup_leaf(q->row[level] + prio, prio,
796 q->ptr[level] + prio,
797 q->last_ptr_id[level] + prio);
798
Linus Torvalds1da177e2005-04-16 15:20:36 -0700799 do {
800next:
Jarek Poplawski512bb432008-12-09 22:35:02 -0800801 if (unlikely(!cl))
Stephen Hemminger87990462006-08-10 23:35:16 -0700802 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700803
804 /* class can be empty - it is unlikely but can be true if leaf
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000805 * qdisc drops packets in enqueue routine or if someone used
806 * graft operation on the leaf since last dequeue;
807 * simply deactivate and skip such class
808 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700809 if (unlikely(cl->un.leaf.q->q.qlen == 0)) {
810 struct htb_class *next;
Stephen Hemminger87990462006-08-10 23:35:16 -0700811 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700812
813 /* row/level might become empty */
814 if ((q->row_mask[level] & (1 << prio)) == 0)
Stephen Hemminger87990462006-08-10 23:35:16 -0700815 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700816
Stephen Hemminger87990462006-08-10 23:35:16 -0700817 next = htb_lookup_leaf(q->row[level] + prio,
818 prio, q->ptr[level] + prio,
819 q->last_ptr_id[level] + prio);
820
821 if (cl == start) /* fix start if we just deleted it */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700822 start = next;
823 cl = next;
824 goto next;
825 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700826
827 skb = cl->un.leaf.q->dequeue(cl->un.leaf.q);
828 if (likely(skb != NULL))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700829 break;
Jarek Poplawski633fe662008-12-03 21:09:10 -0800830
Jarek Poplawskib00355d2009-02-01 01:12:42 -0800831 qdisc_warn_nonwc("htb", cl->un.leaf.q);
Stephen Hemminger87990462006-08-10 23:35:16 -0700832 htb_next_rb_node((level ? cl->parent->un.inner.ptr : q->
833 ptr[0]) + prio);
834 cl = htb_lookup_leaf(q->row[level] + prio, prio,
835 q->ptr[level] + prio,
836 q->last_ptr_id[level] + prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700837
838 } while (cl != start);
839
840 if (likely(skb != NULL)) {
Jussi Kivilinna0abf77e2008-07-20 00:08:27 -0700841 cl->un.leaf.deficit[level] -= qdisc_pkt_len(skb);
842 if (cl->un.leaf.deficit[level] < 0) {
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800843 cl->un.leaf.deficit[level] += cl->quantum;
Stephen Hemminger87990462006-08-10 23:35:16 -0700844 htb_next_rb_node((level ? cl->parent->un.inner.ptr : q->
845 ptr[0]) + prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700846 }
847 /* this used to be after charge_class but this constelation
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000848 * gives us slightly better performance
849 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700850 if (!cl->un.leaf.q->q.qlen)
Stephen Hemminger87990462006-08-10 23:35:16 -0700851 htb_deactivate(q, cl);
Ranjit Manomohanc9726d62007-07-10 22:43:16 -0700852 htb_charge_class(q, cl, level, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700853 }
854 return skb;
855}
856
Linus Torvalds1da177e2005-04-16 15:20:36 -0700857static struct sk_buff *htb_dequeue(struct Qdisc *sch)
858{
859 struct sk_buff *skb = NULL;
860 struct htb_sched *q = qdisc_priv(sch);
861 int level;
Patrick McHardyfb983d42007-03-16 01:22:39 -0700862 psched_time_t next_event;
Jarek Poplawskia73be042009-01-12 21:54:40 -0800863 unsigned long start_at;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700864
865 /* try to dequeue direct packets as high prio (!) to minimize cpu work */
Stephen Hemminger87990462006-08-10 23:35:16 -0700866 skb = __skb_dequeue(&q->direct_queue);
867 if (skb != NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700868 sch->flags &= ~TCQ_F_THROTTLED;
869 sch->q.qlen--;
870 return skb;
871 }
872
Stephen Hemminger87990462006-08-10 23:35:16 -0700873 if (!sch->q.qlen)
874 goto fin;
Patrick McHardy3bebcda2007-03-23 11:29:25 -0700875 q->now = psched_get_time();
Jarek Poplawskia73be042009-01-12 21:54:40 -0800876 start_at = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700877
Patrick McHardyfb983d42007-03-16 01:22:39 -0700878 next_event = q->now + 5 * PSCHED_TICKS_PER_SEC;
Jarek Poplawski633fe662008-12-03 21:09:10 -0800879
Linus Torvalds1da177e2005-04-16 15:20:36 -0700880 for (level = 0; level < TC_HTB_MAXDEPTH; level++) {
881 /* common case optimization - skip event handler quickly */
882 int m;
Patrick McHardyfb983d42007-03-16 01:22:39 -0700883 psched_time_t event;
Stephen Hemminger87990462006-08-10 23:35:16 -0700884
Patrick McHardyfb983d42007-03-16 01:22:39 -0700885 if (q->now >= q->near_ev_cache[level]) {
Jarek Poplawskia73be042009-01-12 21:54:40 -0800886 event = htb_do_events(q, level, start_at);
Patrick McHardy2e4b3b02007-05-23 23:39:54 -0700887 if (!event)
888 event = q->now + PSCHED_TICKS_PER_SEC;
889 q->near_ev_cache[level] = event;
Patrick McHardyfb983d42007-03-16 01:22:39 -0700890 } else
891 event = q->near_ev_cache[level];
892
Jarek Poplawskic0851342009-01-12 21:54:16 -0800893 if (next_event > event)
Patrick McHardyfb983d42007-03-16 01:22:39 -0700894 next_event = event;
895
Linus Torvalds1da177e2005-04-16 15:20:36 -0700896 m = ~q->row_mask[level];
897 while (m != (int)(-1)) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700898 int prio = ffz(m);
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000899
Linus Torvalds1da177e2005-04-16 15:20:36 -0700900 m |= 1 << prio;
Stephen Hemminger87990462006-08-10 23:35:16 -0700901 skb = htb_dequeue_tree(q, prio, level);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700902 if (likely(skb != NULL)) {
903 sch->q.qlen--;
904 sch->flags &= ~TCQ_F_THROTTLED;
905 goto fin;
906 }
907 }
908 }
Patrick McHardyfb983d42007-03-16 01:22:39 -0700909 sch->qstats.overlimits++;
Jarek Poplawski12247362009-02-01 01:13:22 -0800910 if (likely(next_event > q->now))
911 qdisc_watchdog_schedule(&q->watchdog, next_event);
912 else
913 schedule_work(&q->work);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700914fin:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700915 return skb;
916}
917
918/* try to drop from each class (by prio) until one succeed */
Stephen Hemminger87990462006-08-10 23:35:16 -0700919static unsigned int htb_drop(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700920{
921 struct htb_sched *q = qdisc_priv(sch);
922 int prio;
923
924 for (prio = TC_HTB_NUMPRIO - 1; prio >= 0; prio--) {
925 struct list_head *p;
Stephen Hemminger87990462006-08-10 23:35:16 -0700926 list_for_each(p, q->drops + prio) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700927 struct htb_class *cl = list_entry(p, struct htb_class,
928 un.leaf.drop_list);
929 unsigned int len;
Stephen Hemminger87990462006-08-10 23:35:16 -0700930 if (cl->un.leaf.q->ops->drop &&
931 (len = cl->un.leaf.q->ops->drop(cl->un.leaf.q))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700932 sch->q.qlen--;
933 if (!cl->un.leaf.q->q.qlen)
Stephen Hemminger87990462006-08-10 23:35:16 -0700934 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700935 return len;
936 }
937 }
938 }
939 return 0;
940}
941
942/* reset all classes */
943/* always caled under BH & queue lock */
Stephen Hemminger87990462006-08-10 23:35:16 -0700944static void htb_reset(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700945{
946 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700947 struct htb_class *cl;
948 struct hlist_node *n;
949 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700950
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700951 for (i = 0; i < q->clhash.hashsize; i++) {
952 hlist_for_each_entry(cl, n, &q->clhash.hash[i], common.hnode) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700953 if (cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -0700954 memset(&cl->un.inner, 0, sizeof(cl->un.inner));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700955 else {
Stephen Hemminger87990462006-08-10 23:35:16 -0700956 if (cl->un.leaf.q)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700957 qdisc_reset(cl->un.leaf.q);
958 INIT_LIST_HEAD(&cl->un.leaf.drop_list);
959 }
960 cl->prio_activity = 0;
961 cl->cmode = HTB_CAN_SEND;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700962
963 }
964 }
Patrick McHardyfb983d42007-03-16 01:22:39 -0700965 qdisc_watchdog_cancel(&q->watchdog);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700966 __skb_queue_purge(&q->direct_queue);
967 sch->q.qlen = 0;
Stephen Hemminger87990462006-08-10 23:35:16 -0700968 memset(q->row, 0, sizeof(q->row));
969 memset(q->row_mask, 0, sizeof(q->row_mask));
970 memset(q->wait_pq, 0, sizeof(q->wait_pq));
971 memset(q->ptr, 0, sizeof(q->ptr));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700972 for (i = 0; i < TC_HTB_NUMPRIO; i++)
Stephen Hemminger87990462006-08-10 23:35:16 -0700973 INIT_LIST_HEAD(q->drops + i);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700974}
975
Patrick McHardy27a34212008-01-23 20:35:39 -0800976static const struct nla_policy htb_policy[TCA_HTB_MAX + 1] = {
977 [TCA_HTB_PARMS] = { .len = sizeof(struct tc_htb_opt) },
978 [TCA_HTB_INIT] = { .len = sizeof(struct tc_htb_glob) },
979 [TCA_HTB_CTAB] = { .type = NLA_BINARY, .len = TC_RTAB_SIZE },
980 [TCA_HTB_RTAB] = { .type = NLA_BINARY, .len = TC_RTAB_SIZE },
981};
982
Jarek Poplawski12247362009-02-01 01:13:22 -0800983static void htb_work_func(struct work_struct *work)
984{
985 struct htb_sched *q = container_of(work, struct htb_sched, work);
986 struct Qdisc *sch = q->watchdog.qdisc;
987
988 __netif_schedule(qdisc_root(sch));
989}
990
Patrick McHardy1e904742008-01-22 22:11:17 -0800991static int htb_init(struct Qdisc *sch, struct nlattr *opt)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700992{
993 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardy1e904742008-01-22 22:11:17 -0800994 struct nlattr *tb[TCA_HTB_INIT + 1];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700995 struct tc_htb_glob *gopt;
Patrick McHardycee63722008-01-23 20:33:32 -0800996 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700997 int i;
Patrick McHardycee63722008-01-23 20:33:32 -0800998
999 if (!opt)
1000 return -EINVAL;
1001
Patrick McHardy27a34212008-01-23 20:35:39 -08001002 err = nla_parse_nested(tb, TCA_HTB_INIT, opt, htb_policy);
Patrick McHardycee63722008-01-23 20:33:32 -08001003 if (err < 0)
1004 return err;
1005
Patrick McHardy27a34212008-01-23 20:35:39 -08001006 if (tb[TCA_HTB_INIT] == NULL) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001007 pr_err("HTB: hey probably you have bad tc tool ?\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001008 return -EINVAL;
1009 }
Patrick McHardy1e904742008-01-22 22:11:17 -08001010 gopt = nla_data(tb[TCA_HTB_INIT]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001011 if (gopt->version != HTB_VER >> 16) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001012 pr_err("HTB: need tc/htb version %d (minor is %d), you have %d\n",
Stephen Hemminger87990462006-08-10 23:35:16 -07001013 HTB_VER >> 16, HTB_VER & 0xffff, gopt->version);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001014 return -EINVAL;
1015 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001016
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001017 err = qdisc_class_hash_init(&q->clhash);
1018 if (err < 0)
1019 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001020 for (i = 0; i < TC_HTB_NUMPRIO; i++)
Stephen Hemminger87990462006-08-10 23:35:16 -07001021 INIT_LIST_HEAD(q->drops + i);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001022
Patrick McHardyfb983d42007-03-16 01:22:39 -07001023 qdisc_watchdog_init(&q->watchdog, sch);
Jarek Poplawski12247362009-02-01 01:13:22 -08001024 INIT_WORK(&q->work, htb_work_func);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001025 skb_queue_head_init(&q->direct_queue);
1026
David S. Miller5ce2d482008-07-08 17:06:30 -07001027 q->direct_qlen = qdisc_dev(sch)->tx_queue_len;
Stephen Hemminger87990462006-08-10 23:35:16 -07001028 if (q->direct_qlen < 2) /* some devices have zero tx_queue_len */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001029 q->direct_qlen = 2;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001030
Linus Torvalds1da177e2005-04-16 15:20:36 -07001031 if ((q->rate2quantum = gopt->rate2quantum) < 1)
1032 q->rate2quantum = 1;
1033 q->defcls = gopt->defcls;
1034
1035 return 0;
1036}
1037
1038static int htb_dump(struct Qdisc *sch, struct sk_buff *skb)
1039{
Jarek Poplawski102396a2008-08-29 14:21:52 -07001040 spinlock_t *root_lock = qdisc_root_sleeping_lock(sch);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001041 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001042 struct nlattr *nest;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001043 struct tc_htb_glob gopt;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001044
David S. Miller7698b4f2008-07-16 01:42:40 -07001045 spin_lock_bh(root_lock);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001046
1047 gopt.direct_pkts = q->direct_pkts;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001048 gopt.version = HTB_VER;
1049 gopt.rate2quantum = q->rate2quantum;
1050 gopt.defcls = q->defcls;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001051 gopt.debug = 0;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001052
1053 nest = nla_nest_start(skb, TCA_OPTIONS);
1054 if (nest == NULL)
1055 goto nla_put_failure;
Patrick McHardy1e904742008-01-22 22:11:17 -08001056 NLA_PUT(skb, TCA_HTB_INIT, sizeof(gopt), &gopt);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001057 nla_nest_end(skb, nest);
1058
David S. Miller7698b4f2008-07-16 01:42:40 -07001059 spin_unlock_bh(root_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001060 return skb->len;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001061
Patrick McHardy1e904742008-01-22 22:11:17 -08001062nla_put_failure:
David S. Miller7698b4f2008-07-16 01:42:40 -07001063 spin_unlock_bh(root_lock);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001064 nla_nest_cancel(skb, nest);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001065 return -1;
1066}
1067
1068static int htb_dump_class(struct Qdisc *sch, unsigned long arg,
Stephen Hemminger87990462006-08-10 23:35:16 -07001069 struct sk_buff *skb, struct tcmsg *tcm)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001070{
Stephen Hemminger87990462006-08-10 23:35:16 -07001071 struct htb_class *cl = (struct htb_class *)arg;
Jarek Poplawski102396a2008-08-29 14:21:52 -07001072 spinlock_t *root_lock = qdisc_root_sleeping_lock(sch);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001073 struct nlattr *nest;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001074 struct tc_htb_opt opt;
1075
David S. Miller7698b4f2008-07-16 01:42:40 -07001076 spin_lock_bh(root_lock);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001077 tcm->tcm_parent = cl->parent ? cl->parent->common.classid : TC_H_ROOT;
1078 tcm->tcm_handle = cl->common.classid;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001079 if (!cl->level && cl->un.leaf.q)
1080 tcm->tcm_info = cl->un.leaf.q->handle;
1081
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001082 nest = nla_nest_start(skb, TCA_OPTIONS);
1083 if (nest == NULL)
1084 goto nla_put_failure;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001085
Stephen Hemminger87990462006-08-10 23:35:16 -07001086 memset(&opt, 0, sizeof(opt));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001087
Stephen Hemminger87990462006-08-10 23:35:16 -07001088 opt.rate = cl->rate->rate;
1089 opt.buffer = cl->buffer;
1090 opt.ceil = cl->ceil->rate;
1091 opt.cbuffer = cl->cbuffer;
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001092 opt.quantum = cl->quantum;
1093 opt.prio = cl->prio;
Stephen Hemminger87990462006-08-10 23:35:16 -07001094 opt.level = cl->level;
Patrick McHardy1e904742008-01-22 22:11:17 -08001095 NLA_PUT(skb, TCA_HTB_PARMS, sizeof(opt), &opt);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001096
1097 nla_nest_end(skb, nest);
David S. Miller7698b4f2008-07-16 01:42:40 -07001098 spin_unlock_bh(root_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001099 return skb->len;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001100
Patrick McHardy1e904742008-01-22 22:11:17 -08001101nla_put_failure:
David S. Miller7698b4f2008-07-16 01:42:40 -07001102 spin_unlock_bh(root_lock);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001103 nla_nest_cancel(skb, nest);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001104 return -1;
1105}
1106
1107static int
Stephen Hemminger87990462006-08-10 23:35:16 -07001108htb_dump_class_stats(struct Qdisc *sch, unsigned long arg, struct gnet_dump *d)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001109{
Stephen Hemminger87990462006-08-10 23:35:16 -07001110 struct htb_class *cl = (struct htb_class *)arg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001111
Linus Torvalds1da177e2005-04-16 15:20:36 -07001112 if (!cl->level && cl->un.leaf.q)
1113 cl->qstats.qlen = cl->un.leaf.q->q.qlen;
1114 cl->xstats.tokens = cl->tokens;
1115 cl->xstats.ctokens = cl->ctokens;
1116
1117 if (gnet_stats_copy_basic(d, &cl->bstats) < 0 ||
Eric Dumazetd250a5f2009-10-02 10:32:18 +00001118 gnet_stats_copy_rate_est(d, NULL, &cl->rate_est) < 0 ||
Linus Torvalds1da177e2005-04-16 15:20:36 -07001119 gnet_stats_copy_queue(d, &cl->qstats) < 0)
1120 return -1;
1121
1122 return gnet_stats_copy_app(d, &cl->xstats, sizeof(cl->xstats));
1123}
1124
1125static int htb_graft(struct Qdisc *sch, unsigned long arg, struct Qdisc *new,
Stephen Hemminger87990462006-08-10 23:35:16 -07001126 struct Qdisc **old)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001127{
Stephen Hemminger87990462006-08-10 23:35:16 -07001128 struct htb_class *cl = (struct htb_class *)arg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001129
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001130 if (cl->level)
1131 return -EINVAL;
1132 if (new == NULL &&
Changli Gao3511c912010-10-16 13:04:08 +00001133 (new = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops,
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001134 cl->common.classid)) == NULL)
1135 return -ENOBUFS;
1136
1137 sch_tree_lock(sch);
1138 *old = cl->un.leaf.q;
1139 cl->un.leaf.q = new;
1140 if (*old != NULL) {
1141 qdisc_tree_decrease_qlen(*old, (*old)->q.qlen);
1142 qdisc_reset(*old);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001143 }
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001144 sch_tree_unlock(sch);
1145 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001146}
1147
Stephen Hemminger87990462006-08-10 23:35:16 -07001148static struct Qdisc *htb_leaf(struct Qdisc *sch, unsigned long arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001149{
Stephen Hemminger87990462006-08-10 23:35:16 -07001150 struct htb_class *cl = (struct htb_class *)arg;
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001151 return !cl->level ? cl->un.leaf.q : NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001152}
1153
Patrick McHardy256d61b2006-11-29 17:37:05 -08001154static void htb_qlen_notify(struct Qdisc *sch, unsigned long arg)
1155{
1156 struct htb_class *cl = (struct htb_class *)arg;
1157
1158 if (cl->un.leaf.q->q.qlen == 0)
1159 htb_deactivate(qdisc_priv(sch), cl);
1160}
1161
Linus Torvalds1da177e2005-04-16 15:20:36 -07001162static unsigned long htb_get(struct Qdisc *sch, u32 classid)
1163{
Stephen Hemminger87990462006-08-10 23:35:16 -07001164 struct htb_class *cl = htb_find(classid, sch);
1165 if (cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001166 cl->refcnt++;
1167 return (unsigned long)cl;
1168}
1169
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001170static inline int htb_parent_last_child(struct htb_class *cl)
1171{
1172 if (!cl->parent)
1173 /* the root class */
1174 return 0;
Patrick McHardy42077592008-07-05 23:22:53 -07001175 if (cl->parent->children > 1)
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001176 /* not the last child */
1177 return 0;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001178 return 1;
1179}
1180
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001181static void htb_parent_to_leaf(struct htb_sched *q, struct htb_class *cl,
1182 struct Qdisc *new_q)
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001183{
1184 struct htb_class *parent = cl->parent;
1185
Ilpo Järvinen547b7922008-07-25 21:43:18 -07001186 WARN_ON(cl->level || !cl->un.leaf.q || cl->prio_activity);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001187
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001188 if (parent->cmode != HTB_CAN_SEND)
1189 htb_safe_rb_erase(&parent->pq_node, q->wait_pq + parent->level);
1190
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001191 parent->level = 0;
1192 memset(&parent->un.inner, 0, sizeof(parent->un.inner));
1193 INIT_LIST_HEAD(&parent->un.leaf.drop_list);
1194 parent->un.leaf.q = new_q ? new_q : &noop_qdisc;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001195 parent->tokens = parent->buffer;
1196 parent->ctokens = parent->cbuffer;
Patrick McHardy3bebcda2007-03-23 11:29:25 -07001197 parent->t_c = psched_get_time();
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001198 parent->cmode = HTB_CAN_SEND;
1199}
1200
Stephen Hemminger87990462006-08-10 23:35:16 -07001201static void htb_destroy_class(struct Qdisc *sch, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001202{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001203 if (!cl->level) {
Ilpo Järvinen547b7922008-07-25 21:43:18 -07001204 WARN_ON(!cl->un.leaf.q);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001205 qdisc_destroy(cl->un.leaf.q);
1206 }
Patrick McHardyee39e102007-07-02 22:48:13 -07001207 gen_kill_estimator(&cl->bstats, &cl->rate_est);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001208 qdisc_put_rtab(cl->rate);
1209 qdisc_put_rtab(cl->ceil);
Stephen Hemminger87990462006-08-10 23:35:16 -07001210
Patrick McHardyff31ab52008-07-01 19:52:38 -07001211 tcf_destroy_chain(&cl->filter_list);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001212 kfree(cl);
1213}
1214
Stephen Hemminger87990462006-08-10 23:35:16 -07001215static void htb_destroy(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001216{
1217 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001218 struct hlist_node *n, *next;
1219 struct htb_class *cl;
1220 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001221
Jarek Poplawski12247362009-02-01 01:13:22 -08001222 cancel_work_sync(&q->work);
Patrick McHardyfb983d42007-03-16 01:22:39 -07001223 qdisc_watchdog_cancel(&q->watchdog);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001224 /* This line used to be after htb_destroy_class call below
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001225 * and surprisingly it worked in 2.4. But it must precede it
1226 * because filter need its target class alive to be able to call
1227 * unbind_filter on it (without Oops).
1228 */
Patrick McHardyff31ab52008-07-01 19:52:38 -07001229 tcf_destroy_chain(&q->filter_list);
Stephen Hemminger87990462006-08-10 23:35:16 -07001230
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001231 for (i = 0; i < q->clhash.hashsize; i++) {
1232 hlist_for_each_entry(cl, n, &q->clhash.hash[i], common.hnode)
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001233 tcf_destroy_chain(&cl->filter_list);
1234 }
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001235 for (i = 0; i < q->clhash.hashsize; i++) {
1236 hlist_for_each_entry_safe(cl, n, next, &q->clhash.hash[i],
1237 common.hnode)
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001238 htb_destroy_class(sch, cl);
1239 }
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001240 qdisc_class_hash_destroy(&q->clhash);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001241 __skb_queue_purge(&q->direct_queue);
1242}
1243
1244static int htb_delete(struct Qdisc *sch, unsigned long arg)
1245{
1246 struct htb_sched *q = qdisc_priv(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001247 struct htb_class *cl = (struct htb_class *)arg;
Patrick McHardy256d61b2006-11-29 17:37:05 -08001248 unsigned int qlen;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001249 struct Qdisc *new_q = NULL;
1250 int last_child = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001251
1252 // TODO: why don't allow to delete subtree ? references ? does
1253 // tc subsys quarantee us that in htb_destroy it holds no class
1254 // refs so that we can remove children safely there ?
Patrick McHardy42077592008-07-05 23:22:53 -07001255 if (cl->children || cl->filter_cnt)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001256 return -EBUSY;
Stephen Hemminger87990462006-08-10 23:35:16 -07001257
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001258 if (!cl->level && htb_parent_last_child(cl)) {
Changli Gao3511c912010-10-16 13:04:08 +00001259 new_q = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops,
David S. Millerbb949fb2008-07-08 16:55:56 -07001260 cl->parent->common.classid);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001261 last_child = 1;
1262 }
1263
Linus Torvalds1da177e2005-04-16 15:20:36 -07001264 sch_tree_lock(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001265
Patrick McHardy814a175e2006-11-29 17:34:50 -08001266 if (!cl->level) {
Patrick McHardy256d61b2006-11-29 17:37:05 -08001267 qlen = cl->un.leaf.q->q.qlen;
Patrick McHardy814a175e2006-11-29 17:34:50 -08001268 qdisc_reset(cl->un.leaf.q);
Patrick McHardy256d61b2006-11-29 17:37:05 -08001269 qdisc_tree_decrease_qlen(cl->un.leaf.q, qlen);
Patrick McHardy814a175e2006-11-29 17:34:50 -08001270 }
1271
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001272 /* delete from hash and active; remainder in destroy_class */
1273 qdisc_class_hash_remove(&q->clhash, &cl->common);
Jarek Poplawski26b284d2008-08-13 15:16:43 -07001274 if (cl->parent)
1275 cl->parent->children--;
Patrick McHardyc38c83c2007-03-27 14:04:24 -07001276
Linus Torvalds1da177e2005-04-16 15:20:36 -07001277 if (cl->prio_activity)
Stephen Hemminger87990462006-08-10 23:35:16 -07001278 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001279
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001280 if (cl->cmode != HTB_CAN_SEND)
1281 htb_safe_rb_erase(&cl->pq_node, q->wait_pq + cl->level);
1282
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001283 if (last_child)
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001284 htb_parent_to_leaf(q, cl, new_q);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001285
Jarek Poplawski7cd0a632009-03-15 20:00:19 -07001286 BUG_ON(--cl->refcnt == 0);
1287 /*
1288 * This shouldn't happen: we "hold" one cops->get() when called
1289 * from tc_ctl_tclass; the destroy method is done from cops->put().
1290 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001291
1292 sch_tree_unlock(sch);
1293 return 0;
1294}
1295
1296static void htb_put(struct Qdisc *sch, unsigned long arg)
1297{
Stephen Hemminger87990462006-08-10 23:35:16 -07001298 struct htb_class *cl = (struct htb_class *)arg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001299
1300 if (--cl->refcnt == 0)
Stephen Hemminger87990462006-08-10 23:35:16 -07001301 htb_destroy_class(sch, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001302}
1303
Stephen Hemminger87990462006-08-10 23:35:16 -07001304static int htb_change_class(struct Qdisc *sch, u32 classid,
Patrick McHardy1e904742008-01-22 22:11:17 -08001305 u32 parentid, struct nlattr **tca,
Stephen Hemminger87990462006-08-10 23:35:16 -07001306 unsigned long *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001307{
1308 int err = -EINVAL;
1309 struct htb_sched *q = qdisc_priv(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001310 struct htb_class *cl = (struct htb_class *)*arg, *parent;
Patrick McHardy1e904742008-01-22 22:11:17 -08001311 struct nlattr *opt = tca[TCA_OPTIONS];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001312 struct qdisc_rate_table *rtab = NULL, *ctab = NULL;
Changli Gaoe18434c2010-09-30 06:17:44 +00001313 struct nlattr *tb[__TCA_HTB_MAX];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001314 struct tc_htb_opt *hopt;
1315
1316 /* extract all subattrs from opt attr */
Patrick McHardycee63722008-01-23 20:33:32 -08001317 if (!opt)
1318 goto failure;
1319
Changli Gaoe18434c2010-09-30 06:17:44 +00001320 err = nla_parse_nested(tb, TCA_HTB_MAX, opt, htb_policy);
Patrick McHardycee63722008-01-23 20:33:32 -08001321 if (err < 0)
1322 goto failure;
1323
1324 err = -EINVAL;
Patrick McHardy27a34212008-01-23 20:35:39 -08001325 if (tb[TCA_HTB_PARMS] == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001326 goto failure;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001327
Stephen Hemminger87990462006-08-10 23:35:16 -07001328 parent = parentid == TC_H_ROOT ? NULL : htb_find(parentid, sch);
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001329
Patrick McHardy1e904742008-01-22 22:11:17 -08001330 hopt = nla_data(tb[TCA_HTB_PARMS]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001331
Patrick McHardy1e904742008-01-22 22:11:17 -08001332 rtab = qdisc_get_rtab(&hopt->rate, tb[TCA_HTB_RTAB]);
1333 ctab = qdisc_get_rtab(&hopt->ceil, tb[TCA_HTB_CTAB]);
Stephen Hemminger87990462006-08-10 23:35:16 -07001334 if (!rtab || !ctab)
1335 goto failure;
1336
1337 if (!cl) { /* new class */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001338 struct Qdisc *new_q;
Stephen Hemminger3696f622006-08-10 23:36:01 -07001339 int prio;
Patrick McHardyee39e102007-07-02 22:48:13 -07001340 struct {
Patrick McHardy1e904742008-01-22 22:11:17 -08001341 struct nlattr nla;
Patrick McHardyee39e102007-07-02 22:48:13 -07001342 struct gnet_estimator opt;
1343 } est = {
Patrick McHardy1e904742008-01-22 22:11:17 -08001344 .nla = {
1345 .nla_len = nla_attr_size(sizeof(est.opt)),
1346 .nla_type = TCA_RATE,
Patrick McHardyee39e102007-07-02 22:48:13 -07001347 },
1348 .opt = {
1349 /* 4s interval, 16s averaging constant */
1350 .interval = 2,
1351 .ewma_log = 2,
1352 },
1353 };
Stephen Hemminger3696f622006-08-10 23:36:01 -07001354
Linus Torvalds1da177e2005-04-16 15:20:36 -07001355 /* check for valid classid */
Joe Perchesf64f9e72009-11-29 16:55:45 -08001356 if (!classid || TC_H_MAJ(classid ^ sch->handle) ||
1357 htb_find(classid, sch))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001358 goto failure;
1359
1360 /* check maximal depth */
1361 if (parent && parent->parent && parent->parent->level < 2) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001362 pr_err("htb: tree is too deep\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001363 goto failure;
1364 }
1365 err = -ENOBUFS;
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001366 cl = kzalloc(sizeof(*cl), GFP_KERNEL);
1367 if (!cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001368 goto failure;
Stephen Hemminger87990462006-08-10 23:35:16 -07001369
Stephen Hemminger71bcb092008-11-25 21:13:31 -08001370 err = gen_new_estimator(&cl->bstats, &cl->rate_est,
1371 qdisc_root_sleeping_lock(sch),
1372 tca[TCA_RATE] ? : &est.nla);
1373 if (err) {
1374 kfree(cl);
1375 goto failure;
1376 }
1377
Linus Torvalds1da177e2005-04-16 15:20:36 -07001378 cl->refcnt = 1;
Patrick McHardy42077592008-07-05 23:22:53 -07001379 cl->children = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001380 INIT_LIST_HEAD(&cl->un.leaf.drop_list);
Stephen Hemminger3696f622006-08-10 23:36:01 -07001381 RB_CLEAR_NODE(&cl->pq_node);
1382
1383 for (prio = 0; prio < TC_HTB_NUMPRIO; prio++)
1384 RB_CLEAR_NODE(&cl->node[prio]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001385
1386 /* create leaf qdisc early because it uses kmalloc(GFP_KERNEL)
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001387 * so that can't be used inside of sch_tree_lock
1388 * -- thanks to Karlis Peisenieks
1389 */
Changli Gao3511c912010-10-16 13:04:08 +00001390 new_q = qdisc_create_dflt(sch->dev_queue,
David S. Millerbb949fb2008-07-08 16:55:56 -07001391 &pfifo_qdisc_ops, classid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001392 sch_tree_lock(sch);
1393 if (parent && !parent->level) {
Patrick McHardy256d61b2006-11-29 17:37:05 -08001394 unsigned int qlen = parent->un.leaf.q->q.qlen;
1395
Linus Torvalds1da177e2005-04-16 15:20:36 -07001396 /* turn parent into inner node */
Patrick McHardy256d61b2006-11-29 17:37:05 -08001397 qdisc_reset(parent->un.leaf.q);
1398 qdisc_tree_decrease_qlen(parent->un.leaf.q, qlen);
Stephen Hemminger87990462006-08-10 23:35:16 -07001399 qdisc_destroy(parent->un.leaf.q);
1400 if (parent->prio_activity)
1401 htb_deactivate(q, parent);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001402
1403 /* remove from evt list because of level change */
1404 if (parent->cmode != HTB_CAN_SEND) {
Stephen Hemminger3696f622006-08-10 23:36:01 -07001405 htb_safe_rb_erase(&parent->pq_node, q->wait_pq);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001406 parent->cmode = HTB_CAN_SEND;
1407 }
1408 parent->level = (parent->parent ? parent->parent->level
Stephen Hemminger87990462006-08-10 23:35:16 -07001409 : TC_HTB_MAXDEPTH) - 1;
1410 memset(&parent->un.inner, 0, sizeof(parent->un.inner));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001411 }
1412 /* leaf (we) needs elementary qdisc */
1413 cl->un.leaf.q = new_q ? new_q : &noop_qdisc;
1414
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001415 cl->common.classid = classid;
Stephen Hemminger87990462006-08-10 23:35:16 -07001416 cl->parent = parent;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001417
1418 /* set class to be in HTB_CAN_SEND state */
1419 cl->tokens = hopt->buffer;
1420 cl->ctokens = hopt->cbuffer;
Patrick McHardy00c04af2007-03-16 01:23:02 -07001421 cl->mbuffer = 60 * PSCHED_TICKS_PER_SEC; /* 1min */
Patrick McHardy3bebcda2007-03-23 11:29:25 -07001422 cl->t_c = psched_get_time();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001423 cl->cmode = HTB_CAN_SEND;
1424
1425 /* attach to the hash list and parent's family */
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001426 qdisc_class_hash_insert(&q->clhash, &cl->common);
Patrick McHardy42077592008-07-05 23:22:53 -07001427 if (parent)
1428 parent->children++;
Patrick McHardyee39e102007-07-02 22:48:13 -07001429 } else {
Stephen Hemminger71bcb092008-11-25 21:13:31 -08001430 if (tca[TCA_RATE]) {
1431 err = gen_replace_estimator(&cl->bstats, &cl->rate_est,
1432 qdisc_root_sleeping_lock(sch),
1433 tca[TCA_RATE]);
1434 if (err)
1435 return err;
1436 }
Stephen Hemminger87990462006-08-10 23:35:16 -07001437 sch_tree_lock(sch);
Patrick McHardyee39e102007-07-02 22:48:13 -07001438 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001439
1440 /* it used to be a nasty bug here, we have to check that node
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001441 * is really leaf before changing cl->un.leaf !
1442 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001443 if (!cl->level) {
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001444 cl->quantum = rtab->rate.rate / q->rate2quantum;
1445 if (!hopt->quantum && cl->quantum < 1000) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001446 pr_warning(
Stephen Hemminger87990462006-08-10 23:35:16 -07001447 "HTB: quantum of class %X is small. Consider r2q change.\n",
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001448 cl->common.classid);
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001449 cl->quantum = 1000;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001450 }
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001451 if (!hopt->quantum && cl->quantum > 200000) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001452 pr_warning(
Stephen Hemminger87990462006-08-10 23:35:16 -07001453 "HTB: quantum of class %X is big. Consider r2q change.\n",
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001454 cl->common.classid);
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001455 cl->quantum = 200000;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001456 }
1457 if (hopt->quantum)
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001458 cl->quantum = hopt->quantum;
1459 if ((cl->prio = hopt->prio) >= TC_HTB_NUMPRIO)
1460 cl->prio = TC_HTB_NUMPRIO - 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001461 }
1462
1463 cl->buffer = hopt->buffer;
1464 cl->cbuffer = hopt->cbuffer;
Stephen Hemminger87990462006-08-10 23:35:16 -07001465 if (cl->rate)
1466 qdisc_put_rtab(cl->rate);
1467 cl->rate = rtab;
1468 if (cl->ceil)
1469 qdisc_put_rtab(cl->ceil);
1470 cl->ceil = ctab;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001471 sch_tree_unlock(sch);
1472
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001473 qdisc_class_hash_grow(sch, &q->clhash);
1474
Linus Torvalds1da177e2005-04-16 15:20:36 -07001475 *arg = (unsigned long)cl;
1476 return 0;
1477
1478failure:
Stephen Hemminger87990462006-08-10 23:35:16 -07001479 if (rtab)
1480 qdisc_put_rtab(rtab);
1481 if (ctab)
1482 qdisc_put_rtab(ctab);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001483 return err;
1484}
1485
1486static struct tcf_proto **htb_find_tcf(struct Qdisc *sch, unsigned long arg)
1487{
1488 struct htb_sched *q = qdisc_priv(sch);
1489 struct htb_class *cl = (struct htb_class *)arg;
1490 struct tcf_proto **fl = cl ? &cl->filter_list : &q->filter_list;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001491
Linus Torvalds1da177e2005-04-16 15:20:36 -07001492 return fl;
1493}
1494
1495static unsigned long htb_bind_filter(struct Qdisc *sch, unsigned long parent,
Stephen Hemminger87990462006-08-10 23:35:16 -07001496 u32 classid)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001497{
Stephen Hemminger87990462006-08-10 23:35:16 -07001498 struct htb_class *cl = htb_find(classid, sch);
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001499
Linus Torvalds1da177e2005-04-16 15:20:36 -07001500 /*if (cl && !cl->level) return 0;
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001501 * The line above used to be there to prevent attaching filters to
1502 * leaves. But at least tc_index filter uses this just to get class
1503 * for other reasons so that we have to allow for it.
1504 * ----
1505 * 19.6.2002 As Werner explained it is ok - bind filter is just
1506 * another way to "lock" the class - unlike "get" this lock can
1507 * be broken by class during destroy IIUC.
Linus Torvalds1da177e2005-04-16 15:20:36 -07001508 */
Stephen Hemminger87990462006-08-10 23:35:16 -07001509 if (cl)
1510 cl->filter_cnt++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001511 return (unsigned long)cl;
1512}
1513
1514static void htb_unbind_filter(struct Qdisc *sch, unsigned long arg)
1515{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001516 struct htb_class *cl = (struct htb_class *)arg;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001517
Stephen Hemminger87990462006-08-10 23:35:16 -07001518 if (cl)
1519 cl->filter_cnt--;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001520}
1521
1522static void htb_walk(struct Qdisc *sch, struct qdisc_walker *arg)
1523{
1524 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001525 struct htb_class *cl;
1526 struct hlist_node *n;
1527 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001528
1529 if (arg->stop)
1530 return;
1531
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001532 for (i = 0; i < q->clhash.hashsize; i++) {
1533 hlist_for_each_entry(cl, n, &q->clhash.hash[i], common.hnode) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001534 if (arg->count < arg->skip) {
1535 arg->count++;
1536 continue;
1537 }
1538 if (arg->fn(sch, (unsigned long)cl, arg) < 0) {
1539 arg->stop = 1;
1540 return;
1541 }
1542 arg->count++;
1543 }
1544 }
1545}
1546
Eric Dumazet20fea082007-11-14 01:44:41 -08001547static const struct Qdisc_class_ops htb_class_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001548 .graft = htb_graft,
1549 .leaf = htb_leaf,
Patrick McHardy256d61b2006-11-29 17:37:05 -08001550 .qlen_notify = htb_qlen_notify,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001551 .get = htb_get,
1552 .put = htb_put,
1553 .change = htb_change_class,
1554 .delete = htb_delete,
1555 .walk = htb_walk,
1556 .tcf_chain = htb_find_tcf,
1557 .bind_tcf = htb_bind_filter,
1558 .unbind_tcf = htb_unbind_filter,
1559 .dump = htb_dump_class,
1560 .dump_stats = htb_dump_class_stats,
1561};
1562
Eric Dumazet20fea082007-11-14 01:44:41 -08001563static struct Qdisc_ops htb_qdisc_ops __read_mostly = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001564 .cl_ops = &htb_class_ops,
1565 .id = "htb",
1566 .priv_size = sizeof(struct htb_sched),
1567 .enqueue = htb_enqueue,
1568 .dequeue = htb_dequeue,
Jarek Poplawski77be1552008-10-31 00:47:01 -07001569 .peek = qdisc_peek_dequeued,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001570 .drop = htb_drop,
1571 .init = htb_init,
1572 .reset = htb_reset,
1573 .destroy = htb_destroy,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001574 .dump = htb_dump,
1575 .owner = THIS_MODULE,
1576};
1577
1578static int __init htb_module_init(void)
1579{
Stephen Hemminger87990462006-08-10 23:35:16 -07001580 return register_qdisc(&htb_qdisc_ops);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001581}
Stephen Hemminger87990462006-08-10 23:35:16 -07001582static void __exit htb_module_exit(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001583{
Stephen Hemminger87990462006-08-10 23:35:16 -07001584 unregister_qdisc(&htb_qdisc_ops);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001585}
Stephen Hemminger87990462006-08-10 23:35:16 -07001586
Linus Torvalds1da177e2005-04-16 15:20:36 -07001587module_init(htb_module_init)
1588module_exit(htb_module_exit)
1589MODULE_LICENSE("GPL");