blob: 32a80977cf5468b8b5efdd6cc768b8c483dd53e4 [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
Vimalkumar56b765b2012-10-31 06:04:11 +000074struct htb_rate_cfg {
75 u64 rate_bps;
76 u32 mult;
77 u32 shift;
78};
79
Linus Torvalds1da177e2005-04-16 15:20:36 -070080/* interior & leaf nodes; props specific to leaves are marked L: */
Stephen Hemminger87990462006-08-10 23:35:16 -070081struct htb_class {
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -070082 struct Qdisc_class_common common;
Stephen Hemminger87990462006-08-10 23:35:16 -070083 /* general class parameters */
Eric Dumazetc1a8f1f2009-08-16 09:36:49 +000084 struct gnet_stats_basic_packed bstats;
Stephen Hemminger87990462006-08-10 23:35:16 -070085 struct gnet_stats_queue qstats;
86 struct gnet_stats_rate_est rate_est;
87 struct tc_htb_xstats xstats; /* our special stats */
88 int refcnt; /* usage count of this class */
Linus Torvalds1da177e2005-04-16 15:20:36 -070089
Stephen Hemminger87990462006-08-10 23:35:16 -070090 /* topology */
91 int level; /* our level (see above) */
Patrick McHardy42077592008-07-05 23:22:53 -070092 unsigned int children;
Stephen Hemminger87990462006-08-10 23:35:16 -070093 struct htb_class *parent; /* parent class */
Linus Torvalds1da177e2005-04-16 15:20:36 -070094
Jarek Poplawskic19f7a32008-12-03 21:09:45 -080095 int prio; /* these two are used only by leaves... */
96 int quantum; /* but stored for parent-to-leaf return */
97
Stephen Hemminger87990462006-08-10 23:35:16 -070098 union {
99 struct htb_class_leaf {
100 struct Qdisc *q;
Stephen Hemminger87990462006-08-10 23:35:16 -0700101 int deficit[TC_HTB_MAXDEPTH];
102 struct list_head drop_list;
103 } leaf;
104 struct htb_class_inner {
105 struct rb_root feed[TC_HTB_NUMPRIO]; /* feed trees */
106 struct rb_node *ptr[TC_HTB_NUMPRIO]; /* current class ptr */
107 /* When class changes from state 1->2 and disconnects from
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000108 * parent's feed then we lost ptr value and start from the
109 * first child again. Here we store classid of the
110 * last valid ptr (used when ptr is NULL).
111 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700112 u32 last_ptr_id[TC_HTB_NUMPRIO];
113 } inner;
114 } un;
115 struct rb_node node[TC_HTB_NUMPRIO]; /* node for self or feed tree */
116 struct rb_node pq_node; /* node for event queue */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700117 psched_time_t pq_key;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700118
Stephen Hemminger87990462006-08-10 23:35:16 -0700119 int prio_activity; /* for which prios are we active */
120 enum htb_cmode cmode; /* current mode of the class */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700121
Stephen Hemminger87990462006-08-10 23:35:16 -0700122 /* class attached filters */
123 struct tcf_proto *filter_list;
124 int filter_cnt;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700125
Stephen Hemminger87990462006-08-10 23:35:16 -0700126 /* token bucket parameters */
Vimalkumar56b765b2012-10-31 06:04:11 +0000127 struct htb_rate_cfg rate;
128 struct htb_rate_cfg ceil;
129 s64 buffer, cbuffer; /* token bucket depth/rate */
Stephen Hemminger87990462006-08-10 23:35:16 -0700130 psched_tdiff_t mbuffer; /* max wait time */
Vimalkumar56b765b2012-10-31 06:04:11 +0000131 s64 tokens, ctokens; /* current number of tokens */
Stephen Hemminger87990462006-08-10 23:35:16 -0700132 psched_time_t t_c; /* checkpoint time */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700133};
134
Stephen Hemminger87990462006-08-10 23:35:16 -0700135struct htb_sched {
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700136 struct Qdisc_class_hash clhash;
Stephen Hemminger0cef2962006-08-10 23:35:38 -0700137 struct list_head drops[TC_HTB_NUMPRIO];/* active leaves (for drops) */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700138
Stephen Hemminger87990462006-08-10 23:35:16 -0700139 /* self list - roots of self generating tree */
140 struct rb_root row[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO];
141 int row_mask[TC_HTB_MAXDEPTH];
142 struct rb_node *ptr[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO];
143 u32 last_ptr_id[TC_HTB_MAXDEPTH][TC_HTB_NUMPRIO];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700144
Stephen Hemminger87990462006-08-10 23:35:16 -0700145 /* self wait list - roots of wait PQs per row */
146 struct rb_root wait_pq[TC_HTB_MAXDEPTH];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700147
Stephen Hemminger87990462006-08-10 23:35:16 -0700148 /* time of nearest event per level (row) */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700149 psched_time_t near_ev_cache[TC_HTB_MAXDEPTH];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700150
Stephen Hemminger87990462006-08-10 23:35:16 -0700151 int defcls; /* class where unclassified flows go to */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700152
Stephen Hemminger87990462006-08-10 23:35:16 -0700153 /* filters for qdisc itself */
154 struct tcf_proto *filter_list;
Stephen Hemminger87990462006-08-10 23:35:16 -0700155
156 int rate2quantum; /* quant = rate / rate2quantum */
157 psched_time_t now; /* cached dequeue time */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700158 struct qdisc_watchdog watchdog;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700159
Stephen Hemminger87990462006-08-10 23:35:16 -0700160 /* non shaped skbs; let them go directly thru */
161 struct sk_buff_head direct_queue;
162 int direct_qlen; /* max qlen of above */
163
164 long direct_pkts;
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800165
166#define HTB_WARN_TOOMANYEVENTS 0x1
167 unsigned int warned; /* only one warning */
Jarek Poplawski12247362009-02-01 01:13:22 -0800168 struct work_struct work;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700169};
170
Vimalkumar56b765b2012-10-31 06:04:11 +0000171static u64 l2t_ns(struct htb_rate_cfg *r, unsigned int len)
172{
173 return ((u64)len * r->mult) >> r->shift;
174}
175
176static void htb_precompute_ratedata(struct htb_rate_cfg *r)
177{
178 u64 factor;
179 u64 mult;
180 int shift;
181
182 r->shift = 0;
183 r->mult = 1;
184 /*
185 * Calibrate mult, shift so that token counting is accurate
186 * for smallest packet size (64 bytes). Token (time in ns) is
187 * computed as (bytes * 8) * NSEC_PER_SEC / rate_bps. It will
188 * work as long as the smallest packet transfer time can be
189 * accurately represented in nanosec.
190 */
191 if (r->rate_bps > 0) {
192 /*
193 * Higher shift gives better accuracy. Find the largest
194 * shift such that mult fits in 32 bits.
195 */
196 for (shift = 0; shift < 16; shift++) {
197 r->shift = shift;
198 factor = 8LLU * NSEC_PER_SEC * (1 << r->shift);
199 mult = div64_u64(factor, r->rate_bps);
200 if (mult > UINT_MAX)
201 break;
202 }
203
204 r->shift = shift - 1;
205 factor = 8LLU * NSEC_PER_SEC * (1 << r->shift);
206 r->mult = div64_u64(factor, r->rate_bps);
207 }
208}
209
Linus Torvalds1da177e2005-04-16 15:20:36 -0700210/* find class in global hash table using given handle */
Stephen Hemminger87990462006-08-10 23:35:16 -0700211static inline struct htb_class *htb_find(u32 handle, struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700212{
213 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700214 struct Qdisc_class_common *clc;
Stephen Hemminger0cef2962006-08-10 23:35:38 -0700215
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700216 clc = qdisc_class_find(&q->clhash, handle);
217 if (clc == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700218 return NULL;
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700219 return container_of(clc, struct htb_class, common);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700220}
221
222/**
223 * htb_classify - classify a packet into class
224 *
225 * It returns NULL if the packet should be dropped or -1 if the packet
226 * should be passed directly thru. In all other cases leaf class is returned.
227 * We allow direct class selection by classid in priority. The we examine
228 * filters in qdisc and in inner nodes (if higher filter points to the inner
229 * node). If we end up with classid MAJOR:0 we enqueue the skb into special
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900230 * internal fifo (direct). These packets then go directly thru. If we still
Lucas De Marchi25985ed2011-03-30 22:57:33 -0300231 * have no valid leaf we try to use MAJOR:default leaf. It still unsuccessful
Linus Torvalds1da177e2005-04-16 15:20:36 -0700232 * then finish and return direct queue.
233 */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000234#define HTB_DIRECT ((struct htb_class *)-1L)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700235
Stephen Hemminger87990462006-08-10 23:35:16 -0700236static struct htb_class *htb_classify(struct sk_buff *skb, struct Qdisc *sch,
237 int *qerr)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700238{
239 struct htb_sched *q = qdisc_priv(sch);
240 struct htb_class *cl;
241 struct tcf_result res;
242 struct tcf_proto *tcf;
243 int result;
244
245 /* allow to select class by setting skb->priority to valid classid;
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000246 * note that nfmark can be used too by attaching filter fw with no
247 * rules in it
248 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249 if (skb->priority == sch->handle)
Stephen Hemminger87990462006-08-10 23:35:16 -0700250 return HTB_DIRECT; /* X:0 (direct flow) selected */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000251 cl = htb_find(skb->priority, sch);
252 if (cl && cl->level == 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700253 return cl;
254
Jarek Poplawskic27f3392008-08-04 22:39:11 -0700255 *qerr = NET_XMIT_SUCCESS | __NET_XMIT_BYPASS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700256 tcf = q->filter_list;
257 while (tcf && (result = tc_classify(skb, tcf, &res)) >= 0) {
258#ifdef CONFIG_NET_CLS_ACT
259 switch (result) {
260 case TC_ACT_QUEUED:
Stephen Hemminger87990462006-08-10 23:35:16 -0700261 case TC_ACT_STOLEN:
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700262 *qerr = NET_XMIT_SUCCESS | __NET_XMIT_STOLEN;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700263 case TC_ACT_SHOT:
264 return NULL;
265 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700266#endif
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000267 cl = (void *)res.class;
268 if (!cl) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700269 if (res.classid == sch->handle)
Stephen Hemminger87990462006-08-10 23:35:16 -0700270 return HTB_DIRECT; /* X:0 (direct flow) */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000271 cl = htb_find(res.classid, sch);
272 if (!cl)
Stephen Hemminger87990462006-08-10 23:35:16 -0700273 break; /* filter selected invalid classid */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700274 }
275 if (!cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -0700276 return cl; /* we hit leaf; return it */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700277
278 /* we have got inner class; apply inner filter chain */
279 tcf = cl->filter_list;
280 }
281 /* classification failed; try to use default class */
Stephen Hemminger87990462006-08-10 23:35:16 -0700282 cl = htb_find(TC_H_MAKE(TC_H_MAJ(sch->handle), q->defcls), sch);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700283 if (!cl || cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -0700284 return HTB_DIRECT; /* bad default .. this is safe bet */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700285 return cl;
286}
287
Linus Torvalds1da177e2005-04-16 15:20:36 -0700288/**
289 * htb_add_to_id_tree - adds class to the round robin list
290 *
291 * Routine adds class to the list (actually tree) sorted by classid.
292 * Make sure that class is not already on such list for given prio.
293 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700294static void htb_add_to_id_tree(struct rb_root *root,
295 struct htb_class *cl, int prio)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700296{
297 struct rb_node **p = &root->rb_node, *parent = NULL;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700298
Linus Torvalds1da177e2005-04-16 15:20:36 -0700299 while (*p) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700300 struct htb_class *c;
301 parent = *p;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700302 c = rb_entry(parent, struct htb_class, node[prio]);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700303
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700304 if (cl->common.classid > c->common.classid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700305 p = &parent->rb_right;
Stephen Hemminger87990462006-08-10 23:35:16 -0700306 else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700307 p = &parent->rb_left;
308 }
309 rb_link_node(&cl->node[prio], parent, p);
310 rb_insert_color(&cl->node[prio], root);
311}
312
313/**
314 * htb_add_to_wait_tree - adds class to the event queue with delay
315 *
316 * The class is added to priority event queue to indicate that class will
317 * change its mode in cl->pq_key microseconds. Make sure that class is not
318 * already in the queue.
319 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700320static void htb_add_to_wait_tree(struct htb_sched *q,
Vimalkumar56b765b2012-10-31 06:04:11 +0000321 struct htb_class *cl, s64 delay)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700322{
323 struct rb_node **p = &q->wait_pq[cl->level].rb_node, *parent = NULL;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700324
Patrick McHardyfb983d42007-03-16 01:22:39 -0700325 cl->pq_key = q->now + delay;
326 if (cl->pq_key == q->now)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700327 cl->pq_key++;
328
329 /* update the nearest event cache */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700330 if (q->near_ev_cache[cl->level] > cl->pq_key)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700331 q->near_ev_cache[cl->level] = cl->pq_key;
Stephen Hemminger87990462006-08-10 23:35:16 -0700332
Linus Torvalds1da177e2005-04-16 15:20:36 -0700333 while (*p) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700334 struct htb_class *c;
335 parent = *p;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700336 c = rb_entry(parent, struct htb_class, pq_node);
Patrick McHardyfb983d42007-03-16 01:22:39 -0700337 if (cl->pq_key >= c->pq_key)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338 p = &parent->rb_right;
Stephen Hemminger87990462006-08-10 23:35:16 -0700339 else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700340 p = &parent->rb_left;
341 }
342 rb_link_node(&cl->pq_node, parent, p);
343 rb_insert_color(&cl->pq_node, &q->wait_pq[cl->level]);
344}
345
346/**
347 * htb_next_rb_node - finds next node in binary tree
348 *
349 * When we are past last key we return NULL.
350 * Average complexity is 2 steps per call.
351 */
Stephen Hemminger3696f622006-08-10 23:36:01 -0700352static inline void htb_next_rb_node(struct rb_node **n)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700353{
354 *n = rb_next(*n);
355}
356
357/**
358 * htb_add_class_to_row - add class to its row
359 *
360 * The class is added to row at priorities marked in mask.
361 * It does nothing if mask == 0.
362 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700363static inline void htb_add_class_to_row(struct htb_sched *q,
364 struct htb_class *cl, int mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700365{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700366 q->row_mask[cl->level] |= mask;
367 while (mask) {
368 int prio = ffz(~mask);
369 mask &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700370 htb_add_to_id_tree(q->row[cl->level] + prio, cl, prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700371 }
372}
373
Stephen Hemminger3696f622006-08-10 23:36:01 -0700374/* If this triggers, it is a bug in this code, but it need not be fatal */
375static void htb_safe_rb_erase(struct rb_node *rb, struct rb_root *root)
376{
Ismail Donmez81771b32006-10-03 13:49:10 -0700377 if (RB_EMPTY_NODE(rb)) {
Stephen Hemminger3696f622006-08-10 23:36:01 -0700378 WARN_ON(1);
379 } else {
380 rb_erase(rb, root);
381 RB_CLEAR_NODE(rb);
382 }
383}
384
385
Linus Torvalds1da177e2005-04-16 15:20:36 -0700386/**
387 * htb_remove_class_from_row - removes class from its row
388 *
389 * The class is removed from row at priorities marked in mask.
390 * It does nothing if mask == 0.
391 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700392static inline void htb_remove_class_from_row(struct htb_sched *q,
393 struct htb_class *cl, int mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700394{
395 int m = 0;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700396
Linus Torvalds1da177e2005-04-16 15:20:36 -0700397 while (mask) {
398 int prio = ffz(~mask);
Stephen Hemminger3696f622006-08-10 23:36:01 -0700399
Linus Torvalds1da177e2005-04-16 15:20:36 -0700400 mask &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700401 if (q->ptr[cl->level][prio] == cl->node + prio)
402 htb_next_rb_node(q->ptr[cl->level] + prio);
Stephen Hemminger3696f622006-08-10 23:36:01 -0700403
404 htb_safe_rb_erase(cl->node + prio, q->row[cl->level] + prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700405 if (!q->row[cl->level][prio].rb_node)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700406 m |= 1 << prio;
407 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700408 q->row_mask[cl->level] &= ~m;
409}
410
411/**
412 * htb_activate_prios - creates active classe's feed chain
413 *
414 * The class is connected to ancestors and/or appropriate rows
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900415 * for priorities it is participating on. cl->cmode must be new
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416 * (activated) mode. It does nothing if cl->prio_activity == 0.
417 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700418static void htb_activate_prios(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700419{
420 struct htb_class *p = cl->parent;
Stephen Hemminger87990462006-08-10 23:35:16 -0700421 long m, mask = cl->prio_activity;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700422
423 while (cl->cmode == HTB_MAY_BORROW && p && mask) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700424 m = mask;
425 while (m) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700426 int prio = ffz(~m);
427 m &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700428
Linus Torvalds1da177e2005-04-16 15:20:36 -0700429 if (p->un.inner.feed[prio].rb_node)
430 /* parent already has its feed in use so that
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000431 * reset bit in mask as parent is already ok
432 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700433 mask &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700434
435 htb_add_to_id_tree(p->un.inner.feed + prio, cl, prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700436 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700437 p->prio_activity |= mask;
Stephen Hemminger87990462006-08-10 23:35:16 -0700438 cl = p;
439 p = cl->parent;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700440
Linus Torvalds1da177e2005-04-16 15:20:36 -0700441 }
442 if (cl->cmode == HTB_CAN_SEND && mask)
Stephen Hemminger87990462006-08-10 23:35:16 -0700443 htb_add_class_to_row(q, cl, mask);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700444}
445
446/**
447 * htb_deactivate_prios - remove class from feed chain
448 *
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900449 * cl->cmode must represent old mode (before deactivation). It does
Linus Torvalds1da177e2005-04-16 15:20:36 -0700450 * nothing if cl->prio_activity == 0. Class is removed from all feed
451 * chains and rows.
452 */
453static void htb_deactivate_prios(struct htb_sched *q, struct htb_class *cl)
454{
455 struct htb_class *p = cl->parent;
Stephen Hemminger87990462006-08-10 23:35:16 -0700456 long m, mask = cl->prio_activity;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700457
458 while (cl->cmode == HTB_MAY_BORROW && p && mask) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700459 m = mask;
460 mask = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700461 while (m) {
462 int prio = ffz(~m);
463 m &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700464
465 if (p->un.inner.ptr[prio] == cl->node + prio) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700466 /* we are removing child which is pointed to from
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000467 * parent feed - forget the pointer but remember
468 * classid
469 */
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700470 p->un.inner.last_ptr_id[prio] = cl->common.classid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700471 p->un.inner.ptr[prio] = NULL;
472 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700473
Stephen Hemminger3696f622006-08-10 23:36:01 -0700474 htb_safe_rb_erase(cl->node + prio, p->un.inner.feed + prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700475
476 if (!p->un.inner.feed[prio].rb_node)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700477 mask |= 1 << prio;
478 }
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700479
Linus Torvalds1da177e2005-04-16 15:20:36 -0700480 p->prio_activity &= ~mask;
Stephen Hemminger87990462006-08-10 23:35:16 -0700481 cl = p;
482 p = cl->parent;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700483
Linus Torvalds1da177e2005-04-16 15:20:36 -0700484 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700485 if (cl->cmode == HTB_CAN_SEND && mask)
486 htb_remove_class_from_row(q, cl, mask);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700487}
488
Vimalkumar56b765b2012-10-31 06:04:11 +0000489static inline s64 htb_lowater(const struct htb_class *cl)
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700490{
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700491 if (htb_hysteresis)
492 return cl->cmode != HTB_CANT_SEND ? -cl->cbuffer : 0;
493 else
494 return 0;
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700495}
Vimalkumar56b765b2012-10-31 06:04:11 +0000496static inline s64 htb_hiwater(const struct htb_class *cl)
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700497{
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700498 if (htb_hysteresis)
499 return cl->cmode == HTB_CAN_SEND ? -cl->buffer : 0;
500 else
501 return 0;
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700502}
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700503
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700504
Linus Torvalds1da177e2005-04-16 15:20:36 -0700505/**
506 * htb_class_mode - computes and returns current class mode
507 *
508 * It computes cl's mode at time cl->t_c+diff and returns it. If mode
509 * is not HTB_CAN_SEND then cl->pq_key is updated to time difference
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900510 * from now to time when cl will change its state.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700511 * Also it is worth to note that class mode doesn't change simply
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900512 * at cl->{c,}tokens == 0 but there can rather be hysteresis of
Linus Torvalds1da177e2005-04-16 15:20:36 -0700513 * 0 .. -cl->{c,}buffer range. It is meant to limit number of
514 * mode transitions per time unit. The speed gain is about 1/6.
515 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700516static inline enum htb_cmode
Vimalkumar56b765b2012-10-31 06:04:11 +0000517htb_class_mode(struct htb_class *cl, s64 *diff)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700518{
Vimalkumar56b765b2012-10-31 06:04:11 +0000519 s64 toks;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700520
Stephen Hemminger87990462006-08-10 23:35:16 -0700521 if ((toks = (cl->ctokens + *diff)) < htb_lowater(cl)) {
522 *diff = -toks;
523 return HTB_CANT_SEND;
524 }
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700525
Stephen Hemminger87990462006-08-10 23:35:16 -0700526 if ((toks = (cl->tokens + *diff)) >= htb_hiwater(cl))
527 return HTB_CAN_SEND;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700528
Stephen Hemminger87990462006-08-10 23:35:16 -0700529 *diff = -toks;
530 return HTB_MAY_BORROW;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700531}
532
533/**
534 * htb_change_class_mode - changes classe's mode
535 *
536 * This should be the only way how to change classe's mode under normal
537 * cirsumstances. Routine will update feed lists linkage, change mode
538 * and add class to the wait event queue if appropriate. New mode should
539 * be different from old one and cl->pq_key has to be valid if changing
540 * to mode other than HTB_CAN_SEND (see htb_add_to_wait_tree).
541 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700542static void
Vimalkumar56b765b2012-10-31 06:04:11 +0000543htb_change_class_mode(struct htb_sched *q, struct htb_class *cl, s64 *diff)
Stephen Hemminger87990462006-08-10 23:35:16 -0700544{
545 enum htb_cmode new_mode = htb_class_mode(cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700546
547 if (new_mode == cl->cmode)
Stephen Hemminger87990462006-08-10 23:35:16 -0700548 return;
549
550 if (cl->prio_activity) { /* not necessary: speed optimization */
551 if (cl->cmode != HTB_CANT_SEND)
552 htb_deactivate_prios(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700553 cl->cmode = new_mode;
Stephen Hemminger87990462006-08-10 23:35:16 -0700554 if (new_mode != HTB_CANT_SEND)
555 htb_activate_prios(q, cl);
556 } else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700557 cl->cmode = new_mode;
558}
559
560/**
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900561 * htb_activate - inserts leaf cl into appropriate active feeds
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562 *
563 * Routine learns (new) priority of leaf and activates feed chain
564 * for the prio. It can be called on already active leaf safely.
565 * It also adds leaf into droplist.
566 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700567static inline void htb_activate(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700568{
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700569 WARN_ON(cl->level || !cl->un.leaf.q || !cl->un.leaf.q->q.qlen);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700570
Linus Torvalds1da177e2005-04-16 15:20:36 -0700571 if (!cl->prio_activity) {
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800572 cl->prio_activity = 1 << cl->prio;
Stephen Hemminger87990462006-08-10 23:35:16 -0700573 htb_activate_prios(q, cl);
574 list_add_tail(&cl->un.leaf.drop_list,
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800575 q->drops + cl->prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576 }
577}
578
579/**
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900580 * htb_deactivate - remove leaf cl from active feeds
Linus Torvalds1da177e2005-04-16 15:20:36 -0700581 *
582 * Make sure that leaf is active. In the other words it can't be called
583 * with non-active leaf. It also removes class from the drop list.
584 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700585static inline void htb_deactivate(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700586{
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700587 WARN_ON(!cl->prio_activity);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700588
Stephen Hemminger87990462006-08-10 23:35:16 -0700589 htb_deactivate_prios(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700590 cl->prio_activity = 0;
591 list_del_init(&cl->un.leaf.drop_list);
592}
593
594static int htb_enqueue(struct sk_buff *skb, struct Qdisc *sch)
595{
Jarek Poplawskif30ab412008-11-13 22:56:30 -0800596 int uninitialized_var(ret);
Stephen Hemminger87990462006-08-10 23:35:16 -0700597 struct htb_sched *q = qdisc_priv(sch);
598 struct htb_class *cl = htb_classify(skb, sch, &ret);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700599
Stephen Hemminger87990462006-08-10 23:35:16 -0700600 if (cl == HTB_DIRECT) {
601 /* enqueue to helper queue */
602 if (q->direct_queue.qlen < q->direct_qlen) {
603 __skb_queue_tail(&q->direct_queue, skb);
604 q->direct_pkts++;
605 } else {
Eric Dumazet17045752012-05-04 04:37:21 +0000606 return qdisc_drop(skb, sch);
Stephen Hemminger87990462006-08-10 23:35:16 -0700607 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700608#ifdef CONFIG_NET_CLS_ACT
Stephen Hemminger87990462006-08-10 23:35:16 -0700609 } else if (!cl) {
Jarek Poplawskic27f3392008-08-04 22:39:11 -0700610 if (ret & __NET_XMIT_BYPASS)
Stephen Hemminger87990462006-08-10 23:35:16 -0700611 sch->qstats.drops++;
612 kfree_skb(skb);
613 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700614#endif
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700615 } else if ((ret = qdisc_enqueue(skb, cl->un.leaf.q)) != NET_XMIT_SUCCESS) {
616 if (net_xmit_drop_count(ret)) {
617 sch->qstats.drops++;
618 cl->qstats.drops++;
619 }
David S. Miller69747652008-08-17 23:55:36 -0700620 return ret;
Stephen Hemminger87990462006-08-10 23:35:16 -0700621 } else {
Stephen Hemminger87990462006-08-10 23:35:16 -0700622 htb_activate(q, cl);
623 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700624
Stephen Hemminger87990462006-08-10 23:35:16 -0700625 sch->q.qlen++;
Stephen Hemminger87990462006-08-10 23:35:16 -0700626 return NET_XMIT_SUCCESS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700627}
628
Vimalkumar56b765b2012-10-31 06:04:11 +0000629static inline void htb_accnt_tokens(struct htb_class *cl, int bytes, s64 diff)
Jarek Poplawski59e42202008-12-03 21:17:27 -0800630{
Vimalkumar56b765b2012-10-31 06:04:11 +0000631 s64 toks = diff + cl->tokens;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800632
633 if (toks > cl->buffer)
634 toks = cl->buffer;
Vimalkumar56b765b2012-10-31 06:04:11 +0000635 toks -= (s64) l2t_ns(&cl->rate, bytes);
Jarek Poplawski59e42202008-12-03 21:17:27 -0800636 if (toks <= -cl->mbuffer)
637 toks = 1 - cl->mbuffer;
638
639 cl->tokens = toks;
640}
641
Vimalkumar56b765b2012-10-31 06:04:11 +0000642static inline void htb_accnt_ctokens(struct htb_class *cl, int bytes, s64 diff)
Jarek Poplawski59e42202008-12-03 21:17:27 -0800643{
Vimalkumar56b765b2012-10-31 06:04:11 +0000644 s64 toks = diff + cl->ctokens;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800645
646 if (toks > cl->cbuffer)
647 toks = cl->cbuffer;
Vimalkumar56b765b2012-10-31 06:04:11 +0000648 toks -= (s64) l2t_ns(&cl->ceil, bytes);
Jarek Poplawski59e42202008-12-03 21:17:27 -0800649 if (toks <= -cl->mbuffer)
650 toks = 1 - cl->mbuffer;
651
652 cl->ctokens = toks;
653}
654
Linus Torvalds1da177e2005-04-16 15:20:36 -0700655/**
656 * htb_charge_class - charges amount "bytes" to leaf and ancestors
657 *
658 * Routine assumes that packet "bytes" long was dequeued from leaf cl
659 * borrowing from "level". It accounts bytes to ceil leaky bucket for
660 * leaf and all ancestors and to rate bucket for ancestors at levels
661 * "level" and higher. It also handles possible change of mode resulting
662 * from the update. Note that mode can also increase here (MAY_BORROW to
663 * CAN_SEND) because we can use more precise clock that event queue here.
664 * In such case we remove class from event queue first.
665 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700666static void htb_charge_class(struct htb_sched *q, struct htb_class *cl,
Ranjit Manomohanc9726d62007-07-10 22:43:16 -0700667 int level, struct sk_buff *skb)
Stephen Hemminger87990462006-08-10 23:35:16 -0700668{
Jussi Kivilinna0abf77e2008-07-20 00:08:27 -0700669 int bytes = qdisc_pkt_len(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700670 enum htb_cmode old_mode;
Vimalkumar56b765b2012-10-31 06:04:11 +0000671 s64 diff;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700672
673 while (cl) {
Vimalkumar56b765b2012-10-31 06:04:11 +0000674 diff = min_t(s64, q->now - cl->t_c, cl->mbuffer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700675 if (cl->level >= level) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700676 if (cl->level == level)
677 cl->xstats.lends++;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800678 htb_accnt_tokens(cl, bytes, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700679 } else {
680 cl->xstats.borrows++;
Stephen Hemminger87990462006-08-10 23:35:16 -0700681 cl->tokens += diff; /* we moved t_c; update tokens */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700682 }
Jarek Poplawski59e42202008-12-03 21:17:27 -0800683 htb_accnt_ctokens(cl, bytes, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700684 cl->t_c = q->now;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700685
Stephen Hemminger87990462006-08-10 23:35:16 -0700686 old_mode = cl->cmode;
687 diff = 0;
688 htb_change_class_mode(q, cl, &diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700689 if (old_mode != cl->cmode) {
690 if (old_mode != HTB_CAN_SEND)
Stephen Hemminger3696f622006-08-10 23:36:01 -0700691 htb_safe_rb_erase(&cl->pq_node, q->wait_pq + cl->level);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700692 if (cl->cmode != HTB_CAN_SEND)
Stephen Hemminger87990462006-08-10 23:35:16 -0700693 htb_add_to_wait_tree(q, cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700694 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700695
Eric Dumazetbfe0d022011-01-09 08:30:54 +0000696 /* update basic stats except for leaves which are already updated */
697 if (cl->level)
698 bstats_update(&cl->bstats, skb);
699
Linus Torvalds1da177e2005-04-16 15:20:36 -0700700 cl = cl->parent;
701 }
702}
703
704/**
705 * htb_do_events - make mode changes to classes at the level
706 *
Patrick McHardyfb983d42007-03-16 01:22:39 -0700707 * Scans event queue for pending events and applies them. Returns time of
Jarek Poplawski12247362009-02-01 01:13:22 -0800708 * next pending event (0 for no event in pq, q->now for too many events).
Patrick McHardyfb983d42007-03-16 01:22:39 -0700709 * Note: Applied are events whose have cl->pq_key <= q->now.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700710 */
Jarek Poplawskia73be042009-01-12 21:54:40 -0800711static psched_time_t htb_do_events(struct htb_sched *q, int level,
712 unsigned long start)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700713{
Martin Devera8f3ea332008-03-23 22:00:38 -0700714 /* don't run for longer than 2 jiffies; 2 is used instead of
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000715 * 1 to simplify things when jiffy is going to be incremented
716 * too soon
717 */
Jarek Poplawskia73be042009-01-12 21:54:40 -0800718 unsigned long stop_at = start + 2;
Martin Devera8f3ea332008-03-23 22:00:38 -0700719 while (time_before(jiffies, stop_at)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700720 struct htb_class *cl;
Vimalkumar56b765b2012-10-31 06:04:11 +0000721 s64 diff;
Akinbou Mita30bdbe32006-10-12 01:52:05 -0700722 struct rb_node *p = rb_first(&q->wait_pq[level]);
723
Stephen Hemminger87990462006-08-10 23:35:16 -0700724 if (!p)
725 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700726
727 cl = rb_entry(p, struct htb_class, pq_node);
Patrick McHardyfb983d42007-03-16 01:22:39 -0700728 if (cl->pq_key > q->now)
729 return cl->pq_key;
730
Stephen Hemminger3696f622006-08-10 23:36:01 -0700731 htb_safe_rb_erase(p, q->wait_pq + level);
Vimalkumar56b765b2012-10-31 06:04:11 +0000732 diff = min_t(s64, q->now - cl->t_c, cl->mbuffer);
Stephen Hemminger87990462006-08-10 23:35:16 -0700733 htb_change_class_mode(q, cl, &diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700734 if (cl->cmode != HTB_CAN_SEND)
Stephen Hemminger87990462006-08-10 23:35:16 -0700735 htb_add_to_wait_tree(q, cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700736 }
Jarek Poplawski12247362009-02-01 01:13:22 -0800737
738 /* too much load - let's continue after a break for scheduling */
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800739 if (!(q->warned & HTB_WARN_TOOMANYEVENTS)) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000740 pr_warning("htb: too many events!\n");
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800741 q->warned |= HTB_WARN_TOOMANYEVENTS;
742 }
Jarek Poplawski12247362009-02-01 01:13:22 -0800743
744 return q->now;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700745}
746
747/* Returns class->node+prio from id-tree where classe's id is >= id. NULL
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000748 * is no such one exists.
749 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700750static struct rb_node *htb_id_find_next_upper(int prio, struct rb_node *n,
751 u32 id)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700752{
753 struct rb_node *r = NULL;
754 while (n) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700755 struct htb_class *cl =
756 rb_entry(n, struct htb_class, node[prio]);
Stephen Hemminger87990462006-08-10 23:35:16 -0700757
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700758 if (id > cl->common.classid) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700759 n = n->rb_right;
Jarek Poplawski1b5c0072008-12-09 22:34:40 -0800760 } else if (id < cl->common.classid) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700761 r = n;
762 n = n->rb_left;
Jarek Poplawski1b5c0072008-12-09 22:34:40 -0800763 } else {
764 return n;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700765 }
766 }
767 return r;
768}
769
770/**
771 * htb_lookup_leaf - returns next leaf class in DRR order
772 *
773 * Find leaf where current feed pointers points to.
774 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700775static struct htb_class *htb_lookup_leaf(struct rb_root *tree, int prio,
776 struct rb_node **pptr, u32 * pid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700777{
778 int i;
779 struct {
780 struct rb_node *root;
781 struct rb_node **pptr;
782 u32 *pid;
Stephen Hemminger87990462006-08-10 23:35:16 -0700783 } stk[TC_HTB_MAXDEPTH], *sp = stk;
784
Jarek Poplawski512bb432008-12-09 22:35:02 -0800785 BUG_ON(!tree->rb_node);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700786 sp->root = tree->rb_node;
787 sp->pptr = pptr;
788 sp->pid = pid;
789
790 for (i = 0; i < 65535; i++) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700791 if (!*sp->pptr && *sp->pid) {
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900792 /* ptr was invalidated but id is valid - try to recover
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000793 * the original or next ptr
794 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700795 *sp->pptr =
796 htb_id_find_next_upper(prio, sp->root, *sp->pid);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700797 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700798 *sp->pid = 0; /* ptr is valid now so that remove this hint as it
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000799 * can become out of date quickly
800 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700801 if (!*sp->pptr) { /* we are at right end; rewind & go up */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700802 *sp->pptr = sp->root;
Stephen Hemminger87990462006-08-10 23:35:16 -0700803 while ((*sp->pptr)->rb_left)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700804 *sp->pptr = (*sp->pptr)->rb_left;
805 if (sp > stk) {
806 sp--;
Jarek Poplawski512bb432008-12-09 22:35:02 -0800807 if (!*sp->pptr) {
808 WARN_ON(1);
Stephen Hemminger87990462006-08-10 23:35:16 -0700809 return NULL;
Jarek Poplawski512bb432008-12-09 22:35:02 -0800810 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700811 htb_next_rb_node(sp->pptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700812 }
813 } else {
814 struct htb_class *cl;
Stephen Hemminger87990462006-08-10 23:35:16 -0700815 cl = rb_entry(*sp->pptr, struct htb_class, node[prio]);
816 if (!cl->level)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700817 return cl;
818 (++sp)->root = cl->un.inner.feed[prio].rb_node;
Stephen Hemminger87990462006-08-10 23:35:16 -0700819 sp->pptr = cl->un.inner.ptr + prio;
820 sp->pid = cl->un.inner.last_ptr_id + prio;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700821 }
822 }
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700823 WARN_ON(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700824 return NULL;
825}
826
827/* dequeues packet at given priority and level; call only if
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000828 * you are sure that there is active class at prio/level
829 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700830static struct sk_buff *htb_dequeue_tree(struct htb_sched *q, int prio,
831 int level)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700832{
833 struct sk_buff *skb = NULL;
Stephen Hemminger87990462006-08-10 23:35:16 -0700834 struct htb_class *cl, *start;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700835 /* look initial class up in the row */
Stephen Hemminger87990462006-08-10 23:35:16 -0700836 start = cl = htb_lookup_leaf(q->row[level] + prio, prio,
837 q->ptr[level] + prio,
838 q->last_ptr_id[level] + prio);
839
Linus Torvalds1da177e2005-04-16 15:20:36 -0700840 do {
841next:
Jarek Poplawski512bb432008-12-09 22:35:02 -0800842 if (unlikely(!cl))
Stephen Hemminger87990462006-08-10 23:35:16 -0700843 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700844
845 /* class can be empty - it is unlikely but can be true if leaf
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000846 * qdisc drops packets in enqueue routine or if someone used
847 * graft operation on the leaf since last dequeue;
848 * simply deactivate and skip such class
849 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700850 if (unlikely(cl->un.leaf.q->q.qlen == 0)) {
851 struct htb_class *next;
Stephen Hemminger87990462006-08-10 23:35:16 -0700852 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700853
854 /* row/level might become empty */
855 if ((q->row_mask[level] & (1 << prio)) == 0)
Stephen Hemminger87990462006-08-10 23:35:16 -0700856 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700857
Stephen Hemminger87990462006-08-10 23:35:16 -0700858 next = htb_lookup_leaf(q->row[level] + prio,
859 prio, q->ptr[level] + prio,
860 q->last_ptr_id[level] + prio);
861
862 if (cl == start) /* fix start if we just deleted it */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700863 start = next;
864 cl = next;
865 goto next;
866 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700867
868 skb = cl->un.leaf.q->dequeue(cl->un.leaf.q);
869 if (likely(skb != NULL))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700870 break;
Jarek Poplawski633fe662008-12-03 21:09:10 -0800871
Jarek Poplawskib00355d2009-02-01 01:12:42 -0800872 qdisc_warn_nonwc("htb", cl->un.leaf.q);
Stephen Hemminger87990462006-08-10 23:35:16 -0700873 htb_next_rb_node((level ? cl->parent->un.inner.ptr : q->
874 ptr[0]) + prio);
875 cl = htb_lookup_leaf(q->row[level] + prio, prio,
876 q->ptr[level] + prio,
877 q->last_ptr_id[level] + prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700878
879 } while (cl != start);
880
881 if (likely(skb != NULL)) {
Jussi Kivilinna0abf77e2008-07-20 00:08:27 -0700882 cl->un.leaf.deficit[level] -= qdisc_pkt_len(skb);
883 if (cl->un.leaf.deficit[level] < 0) {
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800884 cl->un.leaf.deficit[level] += cl->quantum;
Stephen Hemminger87990462006-08-10 23:35:16 -0700885 htb_next_rb_node((level ? cl->parent->un.inner.ptr : q->
886 ptr[0]) + prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700887 }
888 /* this used to be after charge_class but this constelation
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000889 * gives us slightly better performance
890 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700891 if (!cl->un.leaf.q->q.qlen)
Stephen Hemminger87990462006-08-10 23:35:16 -0700892 htb_deactivate(q, cl);
Ranjit Manomohanc9726d62007-07-10 22:43:16 -0700893 htb_charge_class(q, cl, level, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700894 }
895 return skb;
896}
897
Linus Torvalds1da177e2005-04-16 15:20:36 -0700898static struct sk_buff *htb_dequeue(struct Qdisc *sch)
899{
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800900 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700901 struct htb_sched *q = qdisc_priv(sch);
902 int level;
Patrick McHardyfb983d42007-03-16 01:22:39 -0700903 psched_time_t next_event;
Jarek Poplawskia73be042009-01-12 21:54:40 -0800904 unsigned long start_at;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700905
906 /* try to dequeue direct packets as high prio (!) to minimize cpu work */
Stephen Hemminger87990462006-08-10 23:35:16 -0700907 skb = __skb_dequeue(&q->direct_queue);
908 if (skb != NULL) {
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800909ok:
910 qdisc_bstats_update(sch, skb);
Eric Dumazetfd245a42011-01-20 05:27:16 +0000911 qdisc_unthrottled(sch);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700912 sch->q.qlen--;
913 return skb;
914 }
915
Stephen Hemminger87990462006-08-10 23:35:16 -0700916 if (!sch->q.qlen)
917 goto fin;
Vimalkumar56b765b2012-10-31 06:04:11 +0000918 q->now = ktime_to_ns(ktime_get());
Jarek Poplawskia73be042009-01-12 21:54:40 -0800919 start_at = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700920
Vimalkumar56b765b2012-10-31 06:04:11 +0000921 next_event = q->now + 5 * NSEC_PER_SEC;
Jarek Poplawski633fe662008-12-03 21:09:10 -0800922
Linus Torvalds1da177e2005-04-16 15:20:36 -0700923 for (level = 0; level < TC_HTB_MAXDEPTH; level++) {
924 /* common case optimization - skip event handler quickly */
925 int m;
Patrick McHardyfb983d42007-03-16 01:22:39 -0700926 psched_time_t event;
Stephen Hemminger87990462006-08-10 23:35:16 -0700927
Patrick McHardyfb983d42007-03-16 01:22:39 -0700928 if (q->now >= q->near_ev_cache[level]) {
Jarek Poplawskia73be042009-01-12 21:54:40 -0800929 event = htb_do_events(q, level, start_at);
Patrick McHardy2e4b3b02007-05-23 23:39:54 -0700930 if (!event)
Vimalkumar56b765b2012-10-31 06:04:11 +0000931 event = q->now + NSEC_PER_SEC;
Patrick McHardy2e4b3b02007-05-23 23:39:54 -0700932 q->near_ev_cache[level] = event;
Patrick McHardyfb983d42007-03-16 01:22:39 -0700933 } else
934 event = q->near_ev_cache[level];
935
Jarek Poplawskic0851342009-01-12 21:54:16 -0800936 if (next_event > event)
Patrick McHardyfb983d42007-03-16 01:22:39 -0700937 next_event = event;
938
Linus Torvalds1da177e2005-04-16 15:20:36 -0700939 m = ~q->row_mask[level];
940 while (m != (int)(-1)) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700941 int prio = ffz(m);
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000942
Linus Torvalds1da177e2005-04-16 15:20:36 -0700943 m |= 1 << prio;
Stephen Hemminger87990462006-08-10 23:35:16 -0700944 skb = htb_dequeue_tree(q, prio, level);
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800945 if (likely(skb != NULL))
946 goto ok;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700947 }
948 }
Patrick McHardyfb983d42007-03-16 01:22:39 -0700949 sch->qstats.overlimits++;
Vimalkumar56b765b2012-10-31 06:04:11 +0000950 if (likely(next_event > q->now)) {
951 if (!test_bit(__QDISC_STATE_DEACTIVATED,
952 &qdisc_root_sleeping(q->watchdog.qdisc)->state)) {
953 ktime_t time = ns_to_ktime(next_event);
954 qdisc_throttled(q->watchdog.qdisc);
955 hrtimer_start(&q->watchdog.timer, time,
956 HRTIMER_MODE_ABS);
957 }
958 } else {
Jarek Poplawski12247362009-02-01 01:13:22 -0800959 schedule_work(&q->work);
Vimalkumar56b765b2012-10-31 06:04:11 +0000960 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700961fin:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700962 return skb;
963}
964
965/* try to drop from each class (by prio) until one succeed */
Stephen Hemminger87990462006-08-10 23:35:16 -0700966static unsigned int htb_drop(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700967{
968 struct htb_sched *q = qdisc_priv(sch);
969 int prio;
970
971 for (prio = TC_HTB_NUMPRIO - 1; prio >= 0; prio--) {
972 struct list_head *p;
Stephen Hemminger87990462006-08-10 23:35:16 -0700973 list_for_each(p, q->drops + prio) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700974 struct htb_class *cl = list_entry(p, struct htb_class,
975 un.leaf.drop_list);
976 unsigned int len;
Stephen Hemminger87990462006-08-10 23:35:16 -0700977 if (cl->un.leaf.q->ops->drop &&
978 (len = cl->un.leaf.q->ops->drop(cl->un.leaf.q))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700979 sch->q.qlen--;
980 if (!cl->un.leaf.q->q.qlen)
Stephen Hemminger87990462006-08-10 23:35:16 -0700981 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700982 return len;
983 }
984 }
985 }
986 return 0;
987}
988
989/* reset all classes */
990/* always caled under BH & queue lock */
Stephen Hemminger87990462006-08-10 23:35:16 -0700991static void htb_reset(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700992{
993 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700994 struct htb_class *cl;
995 struct hlist_node *n;
996 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700997
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700998 for (i = 0; i < q->clhash.hashsize; i++) {
999 hlist_for_each_entry(cl, n, &q->clhash.hash[i], common.hnode) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001000 if (cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -07001001 memset(&cl->un.inner, 0, sizeof(cl->un.inner));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001002 else {
Stephen Hemminger87990462006-08-10 23:35:16 -07001003 if (cl->un.leaf.q)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001004 qdisc_reset(cl->un.leaf.q);
1005 INIT_LIST_HEAD(&cl->un.leaf.drop_list);
1006 }
1007 cl->prio_activity = 0;
1008 cl->cmode = HTB_CAN_SEND;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001009
1010 }
1011 }
Patrick McHardyfb983d42007-03-16 01:22:39 -07001012 qdisc_watchdog_cancel(&q->watchdog);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001013 __skb_queue_purge(&q->direct_queue);
1014 sch->q.qlen = 0;
Stephen Hemminger87990462006-08-10 23:35:16 -07001015 memset(q->row, 0, sizeof(q->row));
1016 memset(q->row_mask, 0, sizeof(q->row_mask));
1017 memset(q->wait_pq, 0, sizeof(q->wait_pq));
1018 memset(q->ptr, 0, sizeof(q->ptr));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001019 for (i = 0; i < TC_HTB_NUMPRIO; i++)
Stephen Hemminger87990462006-08-10 23:35:16 -07001020 INIT_LIST_HEAD(q->drops + i);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001021}
1022
Patrick McHardy27a34212008-01-23 20:35:39 -08001023static const struct nla_policy htb_policy[TCA_HTB_MAX + 1] = {
1024 [TCA_HTB_PARMS] = { .len = sizeof(struct tc_htb_opt) },
1025 [TCA_HTB_INIT] = { .len = sizeof(struct tc_htb_glob) },
1026 [TCA_HTB_CTAB] = { .type = NLA_BINARY, .len = TC_RTAB_SIZE },
1027 [TCA_HTB_RTAB] = { .type = NLA_BINARY, .len = TC_RTAB_SIZE },
1028};
1029
Jarek Poplawski12247362009-02-01 01:13:22 -08001030static void htb_work_func(struct work_struct *work)
1031{
1032 struct htb_sched *q = container_of(work, struct htb_sched, work);
1033 struct Qdisc *sch = q->watchdog.qdisc;
1034
1035 __netif_schedule(qdisc_root(sch));
1036}
1037
Patrick McHardy1e904742008-01-22 22:11:17 -08001038static int htb_init(struct Qdisc *sch, struct nlattr *opt)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001039{
1040 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardy1e904742008-01-22 22:11:17 -08001041 struct nlattr *tb[TCA_HTB_INIT + 1];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001042 struct tc_htb_glob *gopt;
Patrick McHardycee63722008-01-23 20:33:32 -08001043 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001044 int i;
Patrick McHardycee63722008-01-23 20:33:32 -08001045
1046 if (!opt)
1047 return -EINVAL;
1048
Patrick McHardy27a34212008-01-23 20:35:39 -08001049 err = nla_parse_nested(tb, TCA_HTB_INIT, opt, htb_policy);
Patrick McHardycee63722008-01-23 20:33:32 -08001050 if (err < 0)
1051 return err;
1052
Patrick McHardy27a34212008-01-23 20:35:39 -08001053 if (tb[TCA_HTB_INIT] == NULL) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001054 pr_err("HTB: hey probably you have bad tc tool ?\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001055 return -EINVAL;
1056 }
Patrick McHardy1e904742008-01-22 22:11:17 -08001057 gopt = nla_data(tb[TCA_HTB_INIT]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001058 if (gopt->version != HTB_VER >> 16) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001059 pr_err("HTB: need tc/htb version %d (minor is %d), you have %d\n",
Stephen Hemminger87990462006-08-10 23:35:16 -07001060 HTB_VER >> 16, HTB_VER & 0xffff, gopt->version);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001061 return -EINVAL;
1062 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001063
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001064 err = qdisc_class_hash_init(&q->clhash);
1065 if (err < 0)
1066 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001067 for (i = 0; i < TC_HTB_NUMPRIO; i++)
Stephen Hemminger87990462006-08-10 23:35:16 -07001068 INIT_LIST_HEAD(q->drops + i);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001069
Patrick McHardyfb983d42007-03-16 01:22:39 -07001070 qdisc_watchdog_init(&q->watchdog, sch);
Jarek Poplawski12247362009-02-01 01:13:22 -08001071 INIT_WORK(&q->work, htb_work_func);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001072 skb_queue_head_init(&q->direct_queue);
1073
David S. Miller5ce2d482008-07-08 17:06:30 -07001074 q->direct_qlen = qdisc_dev(sch)->tx_queue_len;
Stephen Hemminger87990462006-08-10 23:35:16 -07001075 if (q->direct_qlen < 2) /* some devices have zero tx_queue_len */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001076 q->direct_qlen = 2;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001077
Linus Torvalds1da177e2005-04-16 15:20:36 -07001078 if ((q->rate2quantum = gopt->rate2quantum) < 1)
1079 q->rate2quantum = 1;
1080 q->defcls = gopt->defcls;
1081
1082 return 0;
1083}
1084
1085static int htb_dump(struct Qdisc *sch, struct sk_buff *skb)
1086{
Jarek Poplawski102396a2008-08-29 14:21:52 -07001087 spinlock_t *root_lock = qdisc_root_sleeping_lock(sch);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001088 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001089 struct nlattr *nest;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001090 struct tc_htb_glob gopt;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001091
David S. Miller7698b4f2008-07-16 01:42:40 -07001092 spin_lock_bh(root_lock);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001093
1094 gopt.direct_pkts = q->direct_pkts;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001095 gopt.version = HTB_VER;
1096 gopt.rate2quantum = q->rate2quantum;
1097 gopt.defcls = q->defcls;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001098 gopt.debug = 0;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001099
1100 nest = nla_nest_start(skb, TCA_OPTIONS);
1101 if (nest == NULL)
1102 goto nla_put_failure;
David S. Miller1b34ec42012-03-29 05:11:39 -04001103 if (nla_put(skb, TCA_HTB_INIT, sizeof(gopt), &gopt))
1104 goto nla_put_failure;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001105 nla_nest_end(skb, nest);
1106
David S. Miller7698b4f2008-07-16 01:42:40 -07001107 spin_unlock_bh(root_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001108 return skb->len;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001109
Patrick McHardy1e904742008-01-22 22:11:17 -08001110nla_put_failure:
David S. Miller7698b4f2008-07-16 01:42:40 -07001111 spin_unlock_bh(root_lock);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001112 nla_nest_cancel(skb, nest);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001113 return -1;
1114}
1115
1116static int htb_dump_class(struct Qdisc *sch, unsigned long arg,
Stephen Hemminger87990462006-08-10 23:35:16 -07001117 struct sk_buff *skb, struct tcmsg *tcm)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001118{
Stephen Hemminger87990462006-08-10 23:35:16 -07001119 struct htb_class *cl = (struct htb_class *)arg;
Jarek Poplawski102396a2008-08-29 14:21:52 -07001120 spinlock_t *root_lock = qdisc_root_sleeping_lock(sch);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001121 struct nlattr *nest;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001122 struct tc_htb_opt opt;
1123
David S. Miller7698b4f2008-07-16 01:42:40 -07001124 spin_lock_bh(root_lock);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001125 tcm->tcm_parent = cl->parent ? cl->parent->common.classid : TC_H_ROOT;
1126 tcm->tcm_handle = cl->common.classid;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001127 if (!cl->level && cl->un.leaf.q)
1128 tcm->tcm_info = cl->un.leaf.q->handle;
1129
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001130 nest = nla_nest_start(skb, TCA_OPTIONS);
1131 if (nest == NULL)
1132 goto nla_put_failure;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001133
Stephen Hemminger87990462006-08-10 23:35:16 -07001134 memset(&opt, 0, sizeof(opt));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001135
Vimalkumar56b765b2012-10-31 06:04:11 +00001136 opt.rate.rate = cl->rate.rate_bps >> 3;
Stephen Hemminger87990462006-08-10 23:35:16 -07001137 opt.buffer = cl->buffer;
Vimalkumar56b765b2012-10-31 06:04:11 +00001138 opt.ceil.rate = cl->ceil.rate_bps >> 3;
Stephen Hemminger87990462006-08-10 23:35:16 -07001139 opt.cbuffer = cl->cbuffer;
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001140 opt.quantum = cl->quantum;
1141 opt.prio = cl->prio;
Stephen Hemminger87990462006-08-10 23:35:16 -07001142 opt.level = cl->level;
David S. Miller1b34ec42012-03-29 05:11:39 -04001143 if (nla_put(skb, TCA_HTB_PARMS, sizeof(opt), &opt))
1144 goto nla_put_failure;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001145
1146 nla_nest_end(skb, nest);
David S. Miller7698b4f2008-07-16 01:42:40 -07001147 spin_unlock_bh(root_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001148 return skb->len;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001149
Patrick McHardy1e904742008-01-22 22:11:17 -08001150nla_put_failure:
David S. Miller7698b4f2008-07-16 01:42:40 -07001151 spin_unlock_bh(root_lock);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001152 nla_nest_cancel(skb, nest);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001153 return -1;
1154}
1155
1156static int
Stephen Hemminger87990462006-08-10 23:35:16 -07001157htb_dump_class_stats(struct Qdisc *sch, unsigned long arg, struct gnet_dump *d)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001158{
Stephen Hemminger87990462006-08-10 23:35:16 -07001159 struct htb_class *cl = (struct htb_class *)arg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001160
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161 if (!cl->level && cl->un.leaf.q)
1162 cl->qstats.qlen = cl->un.leaf.q->q.qlen;
1163 cl->xstats.tokens = cl->tokens;
1164 cl->xstats.ctokens = cl->ctokens;
1165
1166 if (gnet_stats_copy_basic(d, &cl->bstats) < 0 ||
Eric Dumazetd250a5f2009-10-02 10:32:18 +00001167 gnet_stats_copy_rate_est(d, NULL, &cl->rate_est) < 0 ||
Linus Torvalds1da177e2005-04-16 15:20:36 -07001168 gnet_stats_copy_queue(d, &cl->qstats) < 0)
1169 return -1;
1170
1171 return gnet_stats_copy_app(d, &cl->xstats, sizeof(cl->xstats));
1172}
1173
1174static int htb_graft(struct Qdisc *sch, unsigned long arg, struct Qdisc *new,
Stephen Hemminger87990462006-08-10 23:35:16 -07001175 struct Qdisc **old)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001176{
Stephen Hemminger87990462006-08-10 23:35:16 -07001177 struct htb_class *cl = (struct htb_class *)arg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001178
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001179 if (cl->level)
1180 return -EINVAL;
1181 if (new == NULL &&
Changli Gao3511c912010-10-16 13:04:08 +00001182 (new = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops,
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001183 cl->common.classid)) == NULL)
1184 return -ENOBUFS;
1185
1186 sch_tree_lock(sch);
1187 *old = cl->un.leaf.q;
1188 cl->un.leaf.q = new;
1189 if (*old != NULL) {
1190 qdisc_tree_decrease_qlen(*old, (*old)->q.qlen);
1191 qdisc_reset(*old);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001192 }
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001193 sch_tree_unlock(sch);
1194 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001195}
1196
Stephen Hemminger87990462006-08-10 23:35:16 -07001197static struct Qdisc *htb_leaf(struct Qdisc *sch, unsigned long arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001198{
Stephen Hemminger87990462006-08-10 23:35:16 -07001199 struct htb_class *cl = (struct htb_class *)arg;
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001200 return !cl->level ? cl->un.leaf.q : NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001201}
1202
Patrick McHardy256d61b2006-11-29 17:37:05 -08001203static void htb_qlen_notify(struct Qdisc *sch, unsigned long arg)
1204{
1205 struct htb_class *cl = (struct htb_class *)arg;
1206
1207 if (cl->un.leaf.q->q.qlen == 0)
1208 htb_deactivate(qdisc_priv(sch), cl);
1209}
1210
Linus Torvalds1da177e2005-04-16 15:20:36 -07001211static unsigned long htb_get(struct Qdisc *sch, u32 classid)
1212{
Stephen Hemminger87990462006-08-10 23:35:16 -07001213 struct htb_class *cl = htb_find(classid, sch);
1214 if (cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001215 cl->refcnt++;
1216 return (unsigned long)cl;
1217}
1218
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001219static inline int htb_parent_last_child(struct htb_class *cl)
1220{
1221 if (!cl->parent)
1222 /* the root class */
1223 return 0;
Patrick McHardy42077592008-07-05 23:22:53 -07001224 if (cl->parent->children > 1)
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001225 /* not the last child */
1226 return 0;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001227 return 1;
1228}
1229
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001230static void htb_parent_to_leaf(struct htb_sched *q, struct htb_class *cl,
1231 struct Qdisc *new_q)
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001232{
1233 struct htb_class *parent = cl->parent;
1234
Ilpo Järvinen547b7922008-07-25 21:43:18 -07001235 WARN_ON(cl->level || !cl->un.leaf.q || cl->prio_activity);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001236
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001237 if (parent->cmode != HTB_CAN_SEND)
1238 htb_safe_rb_erase(&parent->pq_node, q->wait_pq + parent->level);
1239
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001240 parent->level = 0;
1241 memset(&parent->un.inner, 0, sizeof(parent->un.inner));
1242 INIT_LIST_HEAD(&parent->un.leaf.drop_list);
1243 parent->un.leaf.q = new_q ? new_q : &noop_qdisc;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001244 parent->tokens = parent->buffer;
1245 parent->ctokens = parent->cbuffer;
Patrick McHardy3bebcda2007-03-23 11:29:25 -07001246 parent->t_c = psched_get_time();
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001247 parent->cmode = HTB_CAN_SEND;
1248}
1249
Stephen Hemminger87990462006-08-10 23:35:16 -07001250static void htb_destroy_class(struct Qdisc *sch, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001251{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001252 if (!cl->level) {
Ilpo Järvinen547b7922008-07-25 21:43:18 -07001253 WARN_ON(!cl->un.leaf.q);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001254 qdisc_destroy(cl->un.leaf.q);
1255 }
Patrick McHardyee39e102007-07-02 22:48:13 -07001256 gen_kill_estimator(&cl->bstats, &cl->rate_est);
Patrick McHardyff31ab52008-07-01 19:52:38 -07001257 tcf_destroy_chain(&cl->filter_list);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001258 kfree(cl);
1259}
1260
Stephen Hemminger87990462006-08-10 23:35:16 -07001261static void htb_destroy(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001262{
1263 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001264 struct hlist_node *n, *next;
1265 struct htb_class *cl;
1266 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001267
Jarek Poplawski12247362009-02-01 01:13:22 -08001268 cancel_work_sync(&q->work);
Patrick McHardyfb983d42007-03-16 01:22:39 -07001269 qdisc_watchdog_cancel(&q->watchdog);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001270 /* This line used to be after htb_destroy_class call below
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001271 * and surprisingly it worked in 2.4. But it must precede it
1272 * because filter need its target class alive to be able to call
1273 * unbind_filter on it (without Oops).
1274 */
Patrick McHardyff31ab52008-07-01 19:52:38 -07001275 tcf_destroy_chain(&q->filter_list);
Stephen Hemminger87990462006-08-10 23:35:16 -07001276
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001277 for (i = 0; i < q->clhash.hashsize; i++) {
1278 hlist_for_each_entry(cl, n, &q->clhash.hash[i], common.hnode)
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001279 tcf_destroy_chain(&cl->filter_list);
1280 }
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001281 for (i = 0; i < q->clhash.hashsize; i++) {
1282 hlist_for_each_entry_safe(cl, n, next, &q->clhash.hash[i],
1283 common.hnode)
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001284 htb_destroy_class(sch, cl);
1285 }
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001286 qdisc_class_hash_destroy(&q->clhash);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001287 __skb_queue_purge(&q->direct_queue);
1288}
1289
1290static int htb_delete(struct Qdisc *sch, unsigned long arg)
1291{
1292 struct htb_sched *q = qdisc_priv(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001293 struct htb_class *cl = (struct htb_class *)arg;
Patrick McHardy256d61b2006-11-29 17:37:05 -08001294 unsigned int qlen;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001295 struct Qdisc *new_q = NULL;
1296 int last_child = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001297
1298 // TODO: why don't allow to delete subtree ? references ? does
1299 // tc subsys quarantee us that in htb_destroy it holds no class
1300 // refs so that we can remove children safely there ?
Patrick McHardy42077592008-07-05 23:22:53 -07001301 if (cl->children || cl->filter_cnt)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001302 return -EBUSY;
Stephen Hemminger87990462006-08-10 23:35:16 -07001303
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001304 if (!cl->level && htb_parent_last_child(cl)) {
Changli Gao3511c912010-10-16 13:04:08 +00001305 new_q = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops,
David S. Millerbb949fb2008-07-08 16:55:56 -07001306 cl->parent->common.classid);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001307 last_child = 1;
1308 }
1309
Linus Torvalds1da177e2005-04-16 15:20:36 -07001310 sch_tree_lock(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001311
Patrick McHardy814a175e2006-11-29 17:34:50 -08001312 if (!cl->level) {
Patrick McHardy256d61b2006-11-29 17:37:05 -08001313 qlen = cl->un.leaf.q->q.qlen;
Patrick McHardy814a175e2006-11-29 17:34:50 -08001314 qdisc_reset(cl->un.leaf.q);
Patrick McHardy256d61b2006-11-29 17:37:05 -08001315 qdisc_tree_decrease_qlen(cl->un.leaf.q, qlen);
Patrick McHardy814a175e2006-11-29 17:34:50 -08001316 }
1317
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001318 /* delete from hash and active; remainder in destroy_class */
1319 qdisc_class_hash_remove(&q->clhash, &cl->common);
Jarek Poplawski26b284d2008-08-13 15:16:43 -07001320 if (cl->parent)
1321 cl->parent->children--;
Patrick McHardyc38c83c2007-03-27 14:04:24 -07001322
Linus Torvalds1da177e2005-04-16 15:20:36 -07001323 if (cl->prio_activity)
Stephen Hemminger87990462006-08-10 23:35:16 -07001324 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001325
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001326 if (cl->cmode != HTB_CAN_SEND)
1327 htb_safe_rb_erase(&cl->pq_node, q->wait_pq + cl->level);
1328
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001329 if (last_child)
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001330 htb_parent_to_leaf(q, cl, new_q);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001331
Jarek Poplawski7cd0a632009-03-15 20:00:19 -07001332 BUG_ON(--cl->refcnt == 0);
1333 /*
1334 * This shouldn't happen: we "hold" one cops->get() when called
1335 * from tc_ctl_tclass; the destroy method is done from cops->put().
1336 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001337
1338 sch_tree_unlock(sch);
1339 return 0;
1340}
1341
1342static void htb_put(struct Qdisc *sch, unsigned long arg)
1343{
Stephen Hemminger87990462006-08-10 23:35:16 -07001344 struct htb_class *cl = (struct htb_class *)arg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001345
1346 if (--cl->refcnt == 0)
Stephen Hemminger87990462006-08-10 23:35:16 -07001347 htb_destroy_class(sch, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001348}
1349
Stephen Hemminger87990462006-08-10 23:35:16 -07001350static int htb_change_class(struct Qdisc *sch, u32 classid,
Patrick McHardy1e904742008-01-22 22:11:17 -08001351 u32 parentid, struct nlattr **tca,
Stephen Hemminger87990462006-08-10 23:35:16 -07001352 unsigned long *arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001353{
1354 int err = -EINVAL;
1355 struct htb_sched *q = qdisc_priv(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001356 struct htb_class *cl = (struct htb_class *)*arg, *parent;
Patrick McHardy1e904742008-01-22 22:11:17 -08001357 struct nlattr *opt = tca[TCA_OPTIONS];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001358 struct qdisc_rate_table *rtab = NULL, *ctab = NULL;
Changli Gaoe18434c2010-09-30 06:17:44 +00001359 struct nlattr *tb[__TCA_HTB_MAX];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001360 struct tc_htb_opt *hopt;
1361
1362 /* extract all subattrs from opt attr */
Patrick McHardycee63722008-01-23 20:33:32 -08001363 if (!opt)
1364 goto failure;
1365
Changli Gaoe18434c2010-09-30 06:17:44 +00001366 err = nla_parse_nested(tb, TCA_HTB_MAX, opt, htb_policy);
Patrick McHardycee63722008-01-23 20:33:32 -08001367 if (err < 0)
1368 goto failure;
1369
1370 err = -EINVAL;
Patrick McHardy27a34212008-01-23 20:35:39 -08001371 if (tb[TCA_HTB_PARMS] == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001372 goto failure;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001373
Stephen Hemminger87990462006-08-10 23:35:16 -07001374 parent = parentid == TC_H_ROOT ? NULL : htb_find(parentid, sch);
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001375
Patrick McHardy1e904742008-01-22 22:11:17 -08001376 hopt = nla_data(tb[TCA_HTB_PARMS]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001377
Patrick McHardy1e904742008-01-22 22:11:17 -08001378 rtab = qdisc_get_rtab(&hopt->rate, tb[TCA_HTB_RTAB]);
1379 ctab = qdisc_get_rtab(&hopt->ceil, tb[TCA_HTB_CTAB]);
Stephen Hemminger87990462006-08-10 23:35:16 -07001380 if (!rtab || !ctab)
1381 goto failure;
1382
1383 if (!cl) { /* new class */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001384 struct Qdisc *new_q;
Stephen Hemminger3696f622006-08-10 23:36:01 -07001385 int prio;
Patrick McHardyee39e102007-07-02 22:48:13 -07001386 struct {
Patrick McHardy1e904742008-01-22 22:11:17 -08001387 struct nlattr nla;
Patrick McHardyee39e102007-07-02 22:48:13 -07001388 struct gnet_estimator opt;
1389 } est = {
Patrick McHardy1e904742008-01-22 22:11:17 -08001390 .nla = {
1391 .nla_len = nla_attr_size(sizeof(est.opt)),
1392 .nla_type = TCA_RATE,
Patrick McHardyee39e102007-07-02 22:48:13 -07001393 },
1394 .opt = {
1395 /* 4s interval, 16s averaging constant */
1396 .interval = 2,
1397 .ewma_log = 2,
1398 },
1399 };
Stephen Hemminger3696f622006-08-10 23:36:01 -07001400
Linus Torvalds1da177e2005-04-16 15:20:36 -07001401 /* check for valid classid */
Joe Perchesf64f9e72009-11-29 16:55:45 -08001402 if (!classid || TC_H_MAJ(classid ^ sch->handle) ||
1403 htb_find(classid, sch))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001404 goto failure;
1405
1406 /* check maximal depth */
1407 if (parent && parent->parent && parent->parent->level < 2) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001408 pr_err("htb: tree is too deep\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001409 goto failure;
1410 }
1411 err = -ENOBUFS;
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001412 cl = kzalloc(sizeof(*cl), GFP_KERNEL);
1413 if (!cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001414 goto failure;
Stephen Hemminger87990462006-08-10 23:35:16 -07001415
Stephen Hemminger71bcb092008-11-25 21:13:31 -08001416 err = gen_new_estimator(&cl->bstats, &cl->rate_est,
1417 qdisc_root_sleeping_lock(sch),
1418 tca[TCA_RATE] ? : &est.nla);
1419 if (err) {
1420 kfree(cl);
1421 goto failure;
1422 }
1423
Linus Torvalds1da177e2005-04-16 15:20:36 -07001424 cl->refcnt = 1;
Patrick McHardy42077592008-07-05 23:22:53 -07001425 cl->children = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001426 INIT_LIST_HEAD(&cl->un.leaf.drop_list);
Stephen Hemminger3696f622006-08-10 23:36:01 -07001427 RB_CLEAR_NODE(&cl->pq_node);
1428
1429 for (prio = 0; prio < TC_HTB_NUMPRIO; prio++)
1430 RB_CLEAR_NODE(&cl->node[prio]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001431
1432 /* create leaf qdisc early because it uses kmalloc(GFP_KERNEL)
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001433 * so that can't be used inside of sch_tree_lock
1434 * -- thanks to Karlis Peisenieks
1435 */
Changli Gao3511c912010-10-16 13:04:08 +00001436 new_q = qdisc_create_dflt(sch->dev_queue,
David S. Millerbb949fb2008-07-08 16:55:56 -07001437 &pfifo_qdisc_ops, classid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001438 sch_tree_lock(sch);
1439 if (parent && !parent->level) {
Patrick McHardy256d61b2006-11-29 17:37:05 -08001440 unsigned int qlen = parent->un.leaf.q->q.qlen;
1441
Linus Torvalds1da177e2005-04-16 15:20:36 -07001442 /* turn parent into inner node */
Patrick McHardy256d61b2006-11-29 17:37:05 -08001443 qdisc_reset(parent->un.leaf.q);
1444 qdisc_tree_decrease_qlen(parent->un.leaf.q, qlen);
Stephen Hemminger87990462006-08-10 23:35:16 -07001445 qdisc_destroy(parent->un.leaf.q);
1446 if (parent->prio_activity)
1447 htb_deactivate(q, parent);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001448
1449 /* remove from evt list because of level change */
1450 if (parent->cmode != HTB_CAN_SEND) {
Stephen Hemminger3696f622006-08-10 23:36:01 -07001451 htb_safe_rb_erase(&parent->pq_node, q->wait_pq);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001452 parent->cmode = HTB_CAN_SEND;
1453 }
1454 parent->level = (parent->parent ? parent->parent->level
Stephen Hemminger87990462006-08-10 23:35:16 -07001455 : TC_HTB_MAXDEPTH) - 1;
1456 memset(&parent->un.inner, 0, sizeof(parent->un.inner));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001457 }
1458 /* leaf (we) needs elementary qdisc */
1459 cl->un.leaf.q = new_q ? new_q : &noop_qdisc;
1460
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001461 cl->common.classid = classid;
Stephen Hemminger87990462006-08-10 23:35:16 -07001462 cl->parent = parent;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001463
1464 /* set class to be in HTB_CAN_SEND state */
1465 cl->tokens = hopt->buffer;
1466 cl->ctokens = hopt->cbuffer;
Patrick McHardy00c04af2007-03-16 01:23:02 -07001467 cl->mbuffer = 60 * PSCHED_TICKS_PER_SEC; /* 1min */
Patrick McHardy3bebcda2007-03-23 11:29:25 -07001468 cl->t_c = psched_get_time();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001469 cl->cmode = HTB_CAN_SEND;
1470
1471 /* attach to the hash list and parent's family */
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001472 qdisc_class_hash_insert(&q->clhash, &cl->common);
Patrick McHardy42077592008-07-05 23:22:53 -07001473 if (parent)
1474 parent->children++;
Patrick McHardyee39e102007-07-02 22:48:13 -07001475 } else {
Stephen Hemminger71bcb092008-11-25 21:13:31 -08001476 if (tca[TCA_RATE]) {
1477 err = gen_replace_estimator(&cl->bstats, &cl->rate_est,
1478 qdisc_root_sleeping_lock(sch),
1479 tca[TCA_RATE]);
1480 if (err)
1481 return err;
1482 }
Stephen Hemminger87990462006-08-10 23:35:16 -07001483 sch_tree_lock(sch);
Patrick McHardyee39e102007-07-02 22:48:13 -07001484 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001485
1486 /* it used to be a nasty bug here, we have to check that node
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001487 * is really leaf before changing cl->un.leaf !
1488 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001489 if (!cl->level) {
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001490 cl->quantum = rtab->rate.rate / q->rate2quantum;
1491 if (!hopt->quantum && cl->quantum < 1000) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001492 pr_warning(
Stephen Hemminger87990462006-08-10 23:35:16 -07001493 "HTB: quantum of class %X is small. Consider r2q change.\n",
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001494 cl->common.classid);
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001495 cl->quantum = 1000;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001496 }
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001497 if (!hopt->quantum && cl->quantum > 200000) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001498 pr_warning(
Stephen Hemminger87990462006-08-10 23:35:16 -07001499 "HTB: quantum of class %X is big. Consider r2q change.\n",
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001500 cl->common.classid);
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001501 cl->quantum = 200000;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001502 }
1503 if (hopt->quantum)
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001504 cl->quantum = hopt->quantum;
1505 if ((cl->prio = hopt->prio) >= TC_HTB_NUMPRIO)
1506 cl->prio = TC_HTB_NUMPRIO - 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001507 }
1508
1509 cl->buffer = hopt->buffer;
1510 cl->cbuffer = hopt->cbuffer;
Vimalkumar56b765b2012-10-31 06:04:11 +00001511
1512 cl->rate.rate_bps = (u64)rtab->rate.rate << 3;
1513 cl->ceil.rate_bps = (u64)ctab->rate.rate << 3;
1514
1515 htb_precompute_ratedata(&cl->rate);
1516 htb_precompute_ratedata(&cl->ceil);
1517
1518 cl->buffer = hopt->buffer << PSCHED_SHIFT;
1519 cl->cbuffer = hopt->buffer << PSCHED_SHIFT;
1520
Linus Torvalds1da177e2005-04-16 15:20:36 -07001521 sch_tree_unlock(sch);
1522
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001523 qdisc_class_hash_grow(sch, &q->clhash);
1524
Linus Torvalds1da177e2005-04-16 15:20:36 -07001525 *arg = (unsigned long)cl;
1526 return 0;
1527
1528failure:
Stephen Hemminger87990462006-08-10 23:35:16 -07001529 if (rtab)
1530 qdisc_put_rtab(rtab);
1531 if (ctab)
1532 qdisc_put_rtab(ctab);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001533 return err;
1534}
1535
1536static struct tcf_proto **htb_find_tcf(struct Qdisc *sch, unsigned long arg)
1537{
1538 struct htb_sched *q = qdisc_priv(sch);
1539 struct htb_class *cl = (struct htb_class *)arg;
1540 struct tcf_proto **fl = cl ? &cl->filter_list : &q->filter_list;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001541
Linus Torvalds1da177e2005-04-16 15:20:36 -07001542 return fl;
1543}
1544
1545static unsigned long htb_bind_filter(struct Qdisc *sch, unsigned long parent,
Stephen Hemminger87990462006-08-10 23:35:16 -07001546 u32 classid)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001547{
Stephen Hemminger87990462006-08-10 23:35:16 -07001548 struct htb_class *cl = htb_find(classid, sch);
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001549
Linus Torvalds1da177e2005-04-16 15:20:36 -07001550 /*if (cl && !cl->level) return 0;
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001551 * The line above used to be there to prevent attaching filters to
1552 * leaves. But at least tc_index filter uses this just to get class
1553 * for other reasons so that we have to allow for it.
1554 * ----
1555 * 19.6.2002 As Werner explained it is ok - bind filter is just
1556 * another way to "lock" the class - unlike "get" this lock can
1557 * be broken by class during destroy IIUC.
Linus Torvalds1da177e2005-04-16 15:20:36 -07001558 */
Stephen Hemminger87990462006-08-10 23:35:16 -07001559 if (cl)
1560 cl->filter_cnt++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001561 return (unsigned long)cl;
1562}
1563
1564static void htb_unbind_filter(struct Qdisc *sch, unsigned long arg)
1565{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001566 struct htb_class *cl = (struct htb_class *)arg;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001567
Stephen Hemminger87990462006-08-10 23:35:16 -07001568 if (cl)
1569 cl->filter_cnt--;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001570}
1571
1572static void htb_walk(struct Qdisc *sch, struct qdisc_walker *arg)
1573{
1574 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001575 struct htb_class *cl;
1576 struct hlist_node *n;
1577 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001578
1579 if (arg->stop)
1580 return;
1581
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001582 for (i = 0; i < q->clhash.hashsize; i++) {
1583 hlist_for_each_entry(cl, n, &q->clhash.hash[i], common.hnode) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001584 if (arg->count < arg->skip) {
1585 arg->count++;
1586 continue;
1587 }
1588 if (arg->fn(sch, (unsigned long)cl, arg) < 0) {
1589 arg->stop = 1;
1590 return;
1591 }
1592 arg->count++;
1593 }
1594 }
1595}
1596
Eric Dumazet20fea082007-11-14 01:44:41 -08001597static const struct Qdisc_class_ops htb_class_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001598 .graft = htb_graft,
1599 .leaf = htb_leaf,
Patrick McHardy256d61b2006-11-29 17:37:05 -08001600 .qlen_notify = htb_qlen_notify,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001601 .get = htb_get,
1602 .put = htb_put,
1603 .change = htb_change_class,
1604 .delete = htb_delete,
1605 .walk = htb_walk,
1606 .tcf_chain = htb_find_tcf,
1607 .bind_tcf = htb_bind_filter,
1608 .unbind_tcf = htb_unbind_filter,
1609 .dump = htb_dump_class,
1610 .dump_stats = htb_dump_class_stats,
1611};
1612
Eric Dumazet20fea082007-11-14 01:44:41 -08001613static struct Qdisc_ops htb_qdisc_ops __read_mostly = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001614 .cl_ops = &htb_class_ops,
1615 .id = "htb",
1616 .priv_size = sizeof(struct htb_sched),
1617 .enqueue = htb_enqueue,
1618 .dequeue = htb_dequeue,
Jarek Poplawski77be1552008-10-31 00:47:01 -07001619 .peek = qdisc_peek_dequeued,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001620 .drop = htb_drop,
1621 .init = htb_init,
1622 .reset = htb_reset,
1623 .destroy = htb_destroy,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001624 .dump = htb_dump,
1625 .owner = THIS_MODULE,
1626};
1627
1628static int __init htb_module_init(void)
1629{
Stephen Hemminger87990462006-08-10 23:35:16 -07001630 return register_qdisc(&htb_qdisc_ops);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001631}
Stephen Hemminger87990462006-08-10 23:35:16 -07001632static void __exit htb_module_exit(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001633{
Stephen Hemminger87990462006-08-10 23:35:16 -07001634 unregister_qdisc(&htb_qdisc_ops);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001635}
Stephen Hemminger87990462006-08-10 23:35:16 -07001636
Linus Torvalds1da177e2005-04-16 15:20:36 -07001637module_init(htb_module_init)
1638module_exit(htb_module_exit)
1639MODULE_LICENSE("GPL");