blob: cd70dbcbd72fdddcda75d98d918f70554af60214 [file] [log] [blame]
Thomas Gleixner2874c5f2019-05-27 08:55:01 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Stephen Hemminger87990462006-08-10 23:35:16 -07002/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07003 * net/sched/sch_htb.c Hierarchical token bucket, feed tree version
4 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 * Authors: Martin Devera, <devik@cdi.cz>
6 *
7 * Credits (in time order) for older HTB versions:
8 * Stef Coene <stef.coene@docum.org>
9 * HTB support at LARTC mailing list
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +090010 * Ondrej Kraus, <krauso@barr.cz>
Linus Torvalds1da177e2005-04-16 15:20:36 -070011 * found missing INIT_QDISC(htb)
12 * Vladimir Smelhaus, Aamer Akhter, Bert Hubert
13 * helped a lot to locate nasty class stall bug
14 * Andi Kleen, Jamal Hadi, Bert Hubert
15 * code review and helpful comments on shaping
16 * Tomasz Wrona, <tw@eter.tym.pl>
17 * created test case so that I was able to fix nasty bug
18 * Wilfried Weissmann
19 * spotted bug in dequeue code and helped with fix
20 * Jiri Fojtasek
21 * fixed requeue routine
22 * and many others. thanks.
Linus Torvalds1da177e2005-04-16 15:20:36 -070023 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070024#include <linux/module.h>
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -070025#include <linux/moduleparam.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070026#include <linux/types.h>
27#include <linux/kernel.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070028#include <linux/string.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070029#include <linux/errno.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070030#include <linux/skbuff.h>
31#include <linux/list.h>
32#include <linux/compiler.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070033#include <linux/rbtree.h>
Jarek Poplawski12247362009-02-01 01:13:22 -080034#include <linux/workqueue.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090035#include <linux/slab.h>
Patrick McHardy0ba48052007-07-02 22:49:07 -070036#include <net/netlink.h>
Jiri Pirko292f1c72013-02-12 00:12:03 +000037#include <net/sch_generic.h>
Patrick McHardy0ba48052007-07-02 22:49:07 -070038#include <net/pkt_sched.h>
Jiri Pirkocf1facd2017-02-09 14:38:56 +010039#include <net/pkt_cls.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070040
41/* HTB algorithm.
42 Author: devik@cdi.cz
43 ========================================================================
44 HTB is like TBF with multiple classes. It is also similar to CBQ because
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +090045 it allows to assign priority to each class in hierarchy.
Linus Torvalds1da177e2005-04-16 15:20:36 -070046 In fact it is another implementation of Floyd's formal sharing.
47
48 Levels:
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +090049 Each class is assigned level. Leaf has ALWAYS level 0 and root
Linus Torvalds1da177e2005-04-16 15:20:36 -070050 classes have level TC_HTB_MAXDEPTH-1. Interior nodes has level
51 one less than their parent.
52*/
53
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -070054static int htb_hysteresis __read_mostly = 0; /* whether to use mode hysteresis for speedup */
Stephen Hemminger87990462006-08-10 23:35:16 -070055#define HTB_VER 0x30011 /* major must be matched with number suplied by TC as version */
Linus Torvalds1da177e2005-04-16 15:20:36 -070056
57#if HTB_VER >> 16 != TC_HTB_PROTOVER
58#error "Mismatched sch_htb.c and pkt_sch.h"
59#endif
60
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -070061/* Module parameter and sysfs export */
62module_param (htb_hysteresis, int, 0640);
63MODULE_PARM_DESC(htb_hysteresis, "Hysteresis mode, less CPU load, less accurate");
64
Eric Dumazet64153ce2013-06-06 14:53:16 -070065static int htb_rate_est = 0; /* htb classes have a default rate estimator */
66module_param(htb_rate_est, int, 0640);
67MODULE_PARM_DESC(htb_rate_est, "setup a default rate estimator (4sec 16sec) for htb classes");
68
Linus Torvalds1da177e2005-04-16 15:20:36 -070069/* used internaly to keep status of single class */
70enum htb_cmode {
Stephen Hemminger87990462006-08-10 23:35:16 -070071 HTB_CANT_SEND, /* class can't send and can't borrow */
72 HTB_MAY_BORROW, /* class can't send but may borrow */
73 HTB_CAN_SEND /* class can send */
Linus Torvalds1da177e2005-04-16 15:20:36 -070074};
75
Eric Dumazetc9364632013-06-15 03:30:10 -070076struct htb_prio {
77 union {
78 struct rb_root row;
79 struct rb_root feed;
80 };
81 struct rb_node *ptr;
82 /* When class changes from state 1->2 and disconnects from
83 * parent's feed then we lost ptr value and start from the
84 * first child again. Here we store classid of the
85 * last valid ptr (used when ptr is NULL).
86 */
87 u32 last_ptr_id;
88};
89
Eric Dumazetca4ec902013-06-13 07:58:30 -070090/* interior & leaf nodes; props specific to leaves are marked L:
91 * To reduce false sharing, place mostly read fields at beginning,
92 * and mostly written ones at the end.
93 */
Stephen Hemminger87990462006-08-10 23:35:16 -070094struct htb_class {
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -070095 struct Qdisc_class_common common;
Eric Dumazetca4ec902013-06-13 07:58:30 -070096 struct psched_ratecfg rate;
97 struct psched_ratecfg ceil;
98 s64 buffer, cbuffer;/* token bucket depth/rate */
99 s64 mbuffer; /* max wait time */
stephen hemmingercbd37552013-08-01 22:32:07 -0700100 u32 prio; /* these two are used only by leaves... */
Eric Dumazetca4ec902013-06-13 07:58:30 -0700101 int quantum; /* but stored for parent-to-leaf return */
102
John Fastabend25d8c0d2014-09-12 20:05:27 -0700103 struct tcf_proto __rcu *filter_list; /* class attached filters */
Jiri Pirko6529eab2017-05-17 11:07:55 +0200104 struct tcf_block *block;
Eric Dumazetca4ec902013-06-13 07:58:30 -0700105 int filter_cnt;
Eric Dumazetca4ec902013-06-13 07:58:30 -0700106
107 int level; /* our level (see above) */
108 unsigned int children;
109 struct htb_class *parent; /* parent class */
110
Eric Dumazet1c0d32f2016-12-04 09:48:16 -0800111 struct net_rate_estimator __rcu *rate_est;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700112
Eric Dumazetca4ec902013-06-13 07:58:30 -0700113 /*
114 * Written often fields
115 */
116 struct gnet_stats_basic_packed bstats;
Eric Dumazetca4ec902013-06-13 07:58:30 -0700117 struct tc_htb_xstats xstats; /* our special stats */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700118
Eric Dumazetca4ec902013-06-13 07:58:30 -0700119 /* token bucket parameters */
120 s64 tokens, ctokens;/* current number of tokens */
121 s64 t_c; /* checkpoint time */
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800122
Stephen Hemminger87990462006-08-10 23:35:16 -0700123 union {
124 struct htb_class_leaf {
Eric Dumazetc9364632013-06-15 03:30:10 -0700125 int deficit[TC_HTB_MAXDEPTH];
126 struct Qdisc *q;
Stephen Hemminger87990462006-08-10 23:35:16 -0700127 } leaf;
128 struct htb_class_inner {
Eric Dumazetc9364632013-06-15 03:30:10 -0700129 struct htb_prio clprio[TC_HTB_NUMPRIO];
Stephen Hemminger87990462006-08-10 23:35:16 -0700130 } inner;
Cong Wang11957be2018-09-07 13:29:14 -0700131 };
Eric Dumazetca4ec902013-06-13 07:58:30 -0700132 s64 pq_key;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700133
Eric Dumazetca4ec902013-06-13 07:58:30 -0700134 int prio_activity; /* for which prios are we active */
135 enum htb_cmode cmode; /* current mode of the class */
136 struct rb_node pq_node; /* node for event queue */
137 struct rb_node node[TC_HTB_NUMPRIO]; /* node for self or feed tree */
Eric Dumazet338ed9b2016-06-21 23:16:51 -0700138
139 unsigned int drops ____cacheline_aligned_in_smp;
Eric Dumazet3c75f6e2017-09-18 12:36:22 -0700140 unsigned int overlimits;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700141};
142
Eric Dumazetc9364632013-06-15 03:30:10 -0700143struct htb_level {
144 struct rb_root wait_pq;
145 struct htb_prio hprio[TC_HTB_NUMPRIO];
146};
147
Stephen Hemminger87990462006-08-10 23:35:16 -0700148struct htb_sched {
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700149 struct Qdisc_class_hash clhash;
Eric Dumazetc9364632013-06-15 03:30:10 -0700150 int defcls; /* class where unclassified flows go to */
151 int rate2quantum; /* quant = rate / rate2quantum */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700152
Stephen Hemminger87990462006-08-10 23:35:16 -0700153 /* filters for qdisc itself */
John Fastabend25d8c0d2014-09-12 20:05:27 -0700154 struct tcf_proto __rcu *filter_list;
Jiri Pirko6529eab2017-05-17 11:07:55 +0200155 struct tcf_block *block;
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800156
157#define HTB_WARN_TOOMANYEVENTS 0x1
Eric Dumazetc9364632013-06-15 03:30:10 -0700158 unsigned int warned; /* only one warning */
159 int direct_qlen;
160 struct work_struct work;
161
162 /* non shaped skbs; let them go directly thru */
Florian Westphal48da34b2016-09-18 00:57:34 +0200163 struct qdisc_skb_head direct_queue;
Cong Wangb3624872019-05-04 11:43:42 -0700164 u32 direct_pkts;
165 u32 overlimits;
Eric Dumazetc9364632013-06-15 03:30:10 -0700166
167 struct qdisc_watchdog watchdog;
168
169 s64 now; /* cached dequeue time */
Eric Dumazetc9364632013-06-15 03:30:10 -0700170
171 /* time of nearest event per level (row) */
172 s64 near_ev_cache[TC_HTB_MAXDEPTH];
173
174 int row_mask[TC_HTB_MAXDEPTH];
175
176 struct htb_level hlevel[TC_HTB_MAXDEPTH];
Linus Torvalds1da177e2005-04-16 15:20:36 -0700177};
178
Linus Torvalds1da177e2005-04-16 15:20:36 -0700179/* find class in global hash table using given handle */
Stephen Hemminger87990462006-08-10 23:35:16 -0700180static inline struct htb_class *htb_find(u32 handle, struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700181{
182 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700183 struct Qdisc_class_common *clc;
Stephen Hemminger0cef2962006-08-10 23:35:38 -0700184
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700185 clc = qdisc_class_find(&q->clhash, handle);
186 if (clc == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700187 return NULL;
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700188 return container_of(clc, struct htb_class, common);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700189}
190
WANG Cong143976c2017-08-24 16:51:29 -0700191static unsigned long htb_search(struct Qdisc *sch, u32 handle)
192{
193 return (unsigned long)htb_find(handle, sch);
194}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700195/**
196 * htb_classify - classify a packet into class
197 *
198 * It returns NULL if the packet should be dropped or -1 if the packet
199 * should be passed directly thru. In all other cases leaf class is returned.
200 * We allow direct class selection by classid in priority. The we examine
201 * filters in qdisc and in inner nodes (if higher filter points to the inner
202 * node). If we end up with classid MAJOR:0 we enqueue the skb into special
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900203 * internal fifo (direct). These packets then go directly thru. If we still
Lucas De Marchi25985ed2011-03-30 22:57:33 -0300204 * have no valid leaf we try to use MAJOR:default leaf. It still unsuccessful
Linus Torvalds1da177e2005-04-16 15:20:36 -0700205 * then finish and return direct queue.
206 */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000207#define HTB_DIRECT ((struct htb_class *)-1L)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700208
Stephen Hemminger87990462006-08-10 23:35:16 -0700209static struct htb_class *htb_classify(struct sk_buff *skb, struct Qdisc *sch,
210 int *qerr)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700211{
212 struct htb_sched *q = qdisc_priv(sch);
213 struct htb_class *cl;
214 struct tcf_result res;
215 struct tcf_proto *tcf;
216 int result;
217
218 /* allow to select class by setting skb->priority to valid classid;
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000219 * note that nfmark can be used too by attaching filter fw with no
220 * rules in it
221 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700222 if (skb->priority == sch->handle)
Stephen Hemminger87990462006-08-10 23:35:16 -0700223 return HTB_DIRECT; /* X:0 (direct flow) selected */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000224 cl = htb_find(skb->priority, sch);
Harry Mason29824312014-01-17 13:22:32 +0000225 if (cl) {
226 if (cl->level == 0)
227 return cl;
228 /* Start with inner filter chain if a non-leaf class is selected */
John Fastabend25d8c0d2014-09-12 20:05:27 -0700229 tcf = rcu_dereference_bh(cl->filter_list);
Harry Mason29824312014-01-17 13:22:32 +0000230 } else {
John Fastabend25d8c0d2014-09-12 20:05:27 -0700231 tcf = rcu_dereference_bh(q->filter_list);
Harry Mason29824312014-01-17 13:22:32 +0000232 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700233
Jarek Poplawskic27f3392008-08-04 22:39:11 -0700234 *qerr = NET_XMIT_SUCCESS | __NET_XMIT_BYPASS;
Jiri Pirko87d83092017-05-17 11:07:54 +0200235 while (tcf && (result = tcf_classify(skb, tcf, &res, false)) >= 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236#ifdef CONFIG_NET_CLS_ACT
237 switch (result) {
238 case TC_ACT_QUEUED:
Stephen Hemminger87990462006-08-10 23:35:16 -0700239 case TC_ACT_STOLEN:
Jiri Pirkoe25ea212017-06-06 14:12:02 +0200240 case TC_ACT_TRAP:
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700241 *qerr = NET_XMIT_SUCCESS | __NET_XMIT_STOLEN;
Gustavo A. R. Silva964201d2020-07-07 12:21:38 -0500242 fallthrough;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243 case TC_ACT_SHOT:
244 return NULL;
245 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700246#endif
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000247 cl = (void *)res.class;
248 if (!cl) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249 if (res.classid == sch->handle)
Stephen Hemminger87990462006-08-10 23:35:16 -0700250 return HTB_DIRECT; /* X:0 (direct flow) */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000251 cl = htb_find(res.classid, sch);
252 if (!cl)
Stephen Hemminger87990462006-08-10 23:35:16 -0700253 break; /* filter selected invalid classid */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700254 }
255 if (!cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -0700256 return cl; /* we hit leaf; return it */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257
258 /* we have got inner class; apply inner filter chain */
John Fastabend25d8c0d2014-09-12 20:05:27 -0700259 tcf = rcu_dereference_bh(cl->filter_list);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700260 }
261 /* classification failed; try to use default class */
Stephen Hemminger87990462006-08-10 23:35:16 -0700262 cl = htb_find(TC_H_MAKE(TC_H_MAJ(sch->handle), q->defcls), sch);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700263 if (!cl || cl->level)
Stephen Hemminger87990462006-08-10 23:35:16 -0700264 return HTB_DIRECT; /* bad default .. this is safe bet */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700265 return cl;
266}
267
Linus Torvalds1da177e2005-04-16 15:20:36 -0700268/**
269 * htb_add_to_id_tree - adds class to the round robin list
270 *
271 * Routine adds class to the list (actually tree) sorted by classid.
272 * Make sure that class is not already on such list for given prio.
273 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700274static void htb_add_to_id_tree(struct rb_root *root,
275 struct htb_class *cl, int prio)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700276{
277 struct rb_node **p = &root->rb_node, *parent = NULL;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700278
Linus Torvalds1da177e2005-04-16 15:20:36 -0700279 while (*p) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700280 struct htb_class *c;
281 parent = *p;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282 c = rb_entry(parent, struct htb_class, node[prio]);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700283
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700284 if (cl->common.classid > c->common.classid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700285 p = &parent->rb_right;
Stephen Hemminger87990462006-08-10 23:35:16 -0700286 else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700287 p = &parent->rb_left;
288 }
289 rb_link_node(&cl->node[prio], parent, p);
290 rb_insert_color(&cl->node[prio], root);
291}
292
293/**
294 * htb_add_to_wait_tree - adds class to the event queue with delay
295 *
296 * The class is added to priority event queue to indicate that class will
297 * change its mode in cl->pq_key microseconds. Make sure that class is not
298 * already in the queue.
299 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700300static void htb_add_to_wait_tree(struct htb_sched *q,
Vimalkumar56b765b2012-10-31 06:04:11 +0000301 struct htb_class *cl, s64 delay)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700302{
Eric Dumazetc9364632013-06-15 03:30:10 -0700303 struct rb_node **p = &q->hlevel[cl->level].wait_pq.rb_node, *parent = NULL;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700304
Patrick McHardyfb983d42007-03-16 01:22:39 -0700305 cl->pq_key = q->now + delay;
306 if (cl->pq_key == q->now)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700307 cl->pq_key++;
308
309 /* update the nearest event cache */
Patrick McHardyfb983d42007-03-16 01:22:39 -0700310 if (q->near_ev_cache[cl->level] > cl->pq_key)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700311 q->near_ev_cache[cl->level] = cl->pq_key;
Stephen Hemminger87990462006-08-10 23:35:16 -0700312
Linus Torvalds1da177e2005-04-16 15:20:36 -0700313 while (*p) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700314 struct htb_class *c;
315 parent = *p;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700316 c = rb_entry(parent, struct htb_class, pq_node);
Patrick McHardyfb983d42007-03-16 01:22:39 -0700317 if (cl->pq_key >= c->pq_key)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318 p = &parent->rb_right;
Stephen Hemminger87990462006-08-10 23:35:16 -0700319 else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700320 p = &parent->rb_left;
321 }
322 rb_link_node(&cl->pq_node, parent, p);
Eric Dumazetc9364632013-06-15 03:30:10 -0700323 rb_insert_color(&cl->pq_node, &q->hlevel[cl->level].wait_pq);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700324}
325
326/**
327 * htb_next_rb_node - finds next node in binary tree
328 *
329 * When we are past last key we return NULL.
330 * Average complexity is 2 steps per call.
331 */
Stephen Hemminger3696f622006-08-10 23:36:01 -0700332static inline void htb_next_rb_node(struct rb_node **n)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700333{
334 *n = rb_next(*n);
335}
336
337/**
338 * htb_add_class_to_row - add class to its row
339 *
340 * The class is added to row at priorities marked in mask.
341 * It does nothing if mask == 0.
342 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700343static inline void htb_add_class_to_row(struct htb_sched *q,
344 struct htb_class *cl, int mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700345{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700346 q->row_mask[cl->level] |= mask;
347 while (mask) {
348 int prio = ffz(~mask);
349 mask &= ~(1 << prio);
Eric Dumazetc9364632013-06-15 03:30:10 -0700350 htb_add_to_id_tree(&q->hlevel[cl->level].hprio[prio].row, cl, prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700351 }
352}
353
Stephen Hemminger3696f622006-08-10 23:36:01 -0700354/* If this triggers, it is a bug in this code, but it need not be fatal */
355static void htb_safe_rb_erase(struct rb_node *rb, struct rb_root *root)
356{
Ismail Donmez81771b32006-10-03 13:49:10 -0700357 if (RB_EMPTY_NODE(rb)) {
Stephen Hemminger3696f622006-08-10 23:36:01 -0700358 WARN_ON(1);
359 } else {
360 rb_erase(rb, root);
361 RB_CLEAR_NODE(rb);
362 }
363}
364
365
Linus Torvalds1da177e2005-04-16 15:20:36 -0700366/**
367 * htb_remove_class_from_row - removes class from its row
368 *
369 * The class is removed from row at priorities marked in mask.
370 * It does nothing if mask == 0.
371 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700372static inline void htb_remove_class_from_row(struct htb_sched *q,
373 struct htb_class *cl, int mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700374{
375 int m = 0;
Eric Dumazetc9364632013-06-15 03:30:10 -0700376 struct htb_level *hlevel = &q->hlevel[cl->level];
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700377
Linus Torvalds1da177e2005-04-16 15:20:36 -0700378 while (mask) {
379 int prio = ffz(~mask);
Eric Dumazetc9364632013-06-15 03:30:10 -0700380 struct htb_prio *hprio = &hlevel->hprio[prio];
Stephen Hemminger3696f622006-08-10 23:36:01 -0700381
Linus Torvalds1da177e2005-04-16 15:20:36 -0700382 mask &= ~(1 << prio);
Eric Dumazetc9364632013-06-15 03:30:10 -0700383 if (hprio->ptr == cl->node + prio)
384 htb_next_rb_node(&hprio->ptr);
Stephen Hemminger3696f622006-08-10 23:36:01 -0700385
Eric Dumazetc9364632013-06-15 03:30:10 -0700386 htb_safe_rb_erase(cl->node + prio, &hprio->row);
387 if (!hprio->row.rb_node)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700388 m |= 1 << prio;
389 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700390 q->row_mask[cl->level] &= ~m;
391}
392
393/**
394 * htb_activate_prios - creates active classe's feed chain
395 *
396 * The class is connected to ancestors and/or appropriate rows
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900397 * for priorities it is participating on. cl->cmode must be new
Linus Torvalds1da177e2005-04-16 15:20:36 -0700398 * (activated) mode. It does nothing if cl->prio_activity == 0.
399 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700400static void htb_activate_prios(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700401{
402 struct htb_class *p = cl->parent;
Stephen Hemminger87990462006-08-10 23:35:16 -0700403 long m, mask = cl->prio_activity;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700404
405 while (cl->cmode == HTB_MAY_BORROW && p && mask) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700406 m = mask;
407 while (m) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700408 int prio = ffz(~m);
409 m &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700410
Cong Wang11957be2018-09-07 13:29:14 -0700411 if (p->inner.clprio[prio].feed.rb_node)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700412 /* parent already has its feed in use so that
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000413 * reset bit in mask as parent is already ok
414 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700415 mask &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700416
Cong Wang11957be2018-09-07 13:29:14 -0700417 htb_add_to_id_tree(&p->inner.clprio[prio].feed, cl, prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700418 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700419 p->prio_activity |= mask;
Stephen Hemminger87990462006-08-10 23:35:16 -0700420 cl = p;
421 p = cl->parent;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700422
Linus Torvalds1da177e2005-04-16 15:20:36 -0700423 }
424 if (cl->cmode == HTB_CAN_SEND && mask)
Stephen Hemminger87990462006-08-10 23:35:16 -0700425 htb_add_class_to_row(q, cl, mask);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700426}
427
428/**
429 * htb_deactivate_prios - remove class from feed chain
430 *
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900431 * cl->cmode must represent old mode (before deactivation). It does
Linus Torvalds1da177e2005-04-16 15:20:36 -0700432 * nothing if cl->prio_activity == 0. Class is removed from all feed
433 * chains and rows.
434 */
435static void htb_deactivate_prios(struct htb_sched *q, struct htb_class *cl)
436{
437 struct htb_class *p = cl->parent;
Stephen Hemminger87990462006-08-10 23:35:16 -0700438 long m, mask = cl->prio_activity;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700439
440 while (cl->cmode == HTB_MAY_BORROW && p && mask) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700441 m = mask;
442 mask = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700443 while (m) {
444 int prio = ffz(~m);
445 m &= ~(1 << prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700446
Cong Wang11957be2018-09-07 13:29:14 -0700447 if (p->inner.clprio[prio].ptr == cl->node + prio) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700448 /* we are removing child which is pointed to from
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000449 * parent feed - forget the pointer but remember
450 * classid
451 */
Cong Wang11957be2018-09-07 13:29:14 -0700452 p->inner.clprio[prio].last_ptr_id = cl->common.classid;
453 p->inner.clprio[prio].ptr = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700454 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700455
Eric Dumazetc9364632013-06-15 03:30:10 -0700456 htb_safe_rb_erase(cl->node + prio,
Cong Wang11957be2018-09-07 13:29:14 -0700457 &p->inner.clprio[prio].feed);
Stephen Hemminger87990462006-08-10 23:35:16 -0700458
Cong Wang11957be2018-09-07 13:29:14 -0700459 if (!p->inner.clprio[prio].feed.rb_node)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700460 mask |= 1 << prio;
461 }
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700462
Linus Torvalds1da177e2005-04-16 15:20:36 -0700463 p->prio_activity &= ~mask;
Stephen Hemminger87990462006-08-10 23:35:16 -0700464 cl = p;
465 p = cl->parent;
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700466
Linus Torvalds1da177e2005-04-16 15:20:36 -0700467 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700468 if (cl->cmode == HTB_CAN_SEND && mask)
469 htb_remove_class_from_row(q, cl, mask);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700470}
471
Vimalkumar56b765b2012-10-31 06:04:11 +0000472static inline s64 htb_lowater(const struct htb_class *cl)
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700473{
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700474 if (htb_hysteresis)
475 return cl->cmode != HTB_CANT_SEND ? -cl->cbuffer : 0;
476 else
477 return 0;
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700478}
Vimalkumar56b765b2012-10-31 06:04:11 +0000479static inline s64 htb_hiwater(const struct htb_class *cl)
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700480{
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700481 if (htb_hysteresis)
482 return cl->cmode == HTB_CAN_SEND ? -cl->buffer : 0;
483 else
484 return 0;
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700485}
Jesper Dangaard Brouer47083fc2008-06-16 16:39:32 -0700486
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700487
Linus Torvalds1da177e2005-04-16 15:20:36 -0700488/**
489 * htb_class_mode - computes and returns current class mode
490 *
491 * It computes cl's mode at time cl->t_c+diff and returns it. If mode
492 * is not HTB_CAN_SEND then cl->pq_key is updated to time difference
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900493 * from now to time when cl will change its state.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700494 * Also it is worth to note that class mode doesn't change simply
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900495 * at cl->{c,}tokens == 0 but there can rather be hysteresis of
Linus Torvalds1da177e2005-04-16 15:20:36 -0700496 * 0 .. -cl->{c,}buffer range. It is meant to limit number of
497 * mode transitions per time unit. The speed gain is about 1/6.
498 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700499static inline enum htb_cmode
Vimalkumar56b765b2012-10-31 06:04:11 +0000500htb_class_mode(struct htb_class *cl, s64 *diff)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700501{
Vimalkumar56b765b2012-10-31 06:04:11 +0000502 s64 toks;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700503
Stephen Hemminger87990462006-08-10 23:35:16 -0700504 if ((toks = (cl->ctokens + *diff)) < htb_lowater(cl)) {
505 *diff = -toks;
506 return HTB_CANT_SEND;
507 }
Stephen Hemminger18a63e82006-08-10 23:34:02 -0700508
Stephen Hemminger87990462006-08-10 23:35:16 -0700509 if ((toks = (cl->tokens + *diff)) >= htb_hiwater(cl))
510 return HTB_CAN_SEND;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700511
Stephen Hemminger87990462006-08-10 23:35:16 -0700512 *diff = -toks;
513 return HTB_MAY_BORROW;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700514}
515
516/**
517 * htb_change_class_mode - changes classe's mode
518 *
519 * This should be the only way how to change classe's mode under normal
520 * cirsumstances. Routine will update feed lists linkage, change mode
521 * and add class to the wait event queue if appropriate. New mode should
522 * be different from old one and cl->pq_key has to be valid if changing
523 * to mode other than HTB_CAN_SEND (see htb_add_to_wait_tree).
524 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700525static void
Vimalkumar56b765b2012-10-31 06:04:11 +0000526htb_change_class_mode(struct htb_sched *q, struct htb_class *cl, s64 *diff)
Stephen Hemminger87990462006-08-10 23:35:16 -0700527{
528 enum htb_cmode new_mode = htb_class_mode(cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700529
530 if (new_mode == cl->cmode)
Stephen Hemminger87990462006-08-10 23:35:16 -0700531 return;
532
Cong Wangb3624872019-05-04 11:43:42 -0700533 if (new_mode == HTB_CANT_SEND) {
Eric Dumazet3c75f6e2017-09-18 12:36:22 -0700534 cl->overlimits++;
Cong Wangb3624872019-05-04 11:43:42 -0700535 q->overlimits++;
536 }
Eric Dumazet3c75f6e2017-09-18 12:36:22 -0700537
Stephen Hemminger87990462006-08-10 23:35:16 -0700538 if (cl->prio_activity) { /* not necessary: speed optimization */
539 if (cl->cmode != HTB_CANT_SEND)
540 htb_deactivate_prios(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700541 cl->cmode = new_mode;
Stephen Hemminger87990462006-08-10 23:35:16 -0700542 if (new_mode != HTB_CANT_SEND)
543 htb_activate_prios(q, cl);
544 } else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545 cl->cmode = new_mode;
546}
547
548/**
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900549 * htb_activate - inserts leaf cl into appropriate active feeds
Linus Torvalds1da177e2005-04-16 15:20:36 -0700550 *
551 * Routine learns (new) priority of leaf and activates feed chain
552 * for the prio. It can be called on already active leaf safely.
553 * It also adds leaf into droplist.
554 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700555static inline void htb_activate(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700556{
Cong Wang11957be2018-09-07 13:29:14 -0700557 WARN_ON(cl->level || !cl->leaf.q || !cl->leaf.q->q.qlen);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700558
Linus Torvalds1da177e2005-04-16 15:20:36 -0700559 if (!cl->prio_activity) {
Jarek Poplawskic19f7a32008-12-03 21:09:45 -0800560 cl->prio_activity = 1 << cl->prio;
Stephen Hemminger87990462006-08-10 23:35:16 -0700561 htb_activate_prios(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562 }
563}
564
565/**
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900566 * htb_deactivate - remove leaf cl from active feeds
Linus Torvalds1da177e2005-04-16 15:20:36 -0700567 *
568 * Make sure that leaf is active. In the other words it can't be called
569 * with non-active leaf. It also removes class from the drop list.
570 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700571static inline void htb_deactivate(struct htb_sched *q, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700572{
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700573 WARN_ON(!cl->prio_activity);
Stephen Hemminger3bf72952006-08-10 23:31:08 -0700574
Stephen Hemminger87990462006-08-10 23:35:16 -0700575 htb_deactivate_prios(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576 cl->prio_activity = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700577}
578
Eric Dumazet520ac302016-06-21 23:16:49 -0700579static int htb_enqueue(struct sk_buff *skb, struct Qdisc *sch,
580 struct sk_buff **to_free)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700581{
Kees Cook3f649ab2020-06-03 13:09:38 -0700582 int ret;
Toke Høiland-Jørgensenf6bab192019-01-09 17:09:42 +0100583 unsigned int len = qdisc_pkt_len(skb);
Stephen Hemminger87990462006-08-10 23:35:16 -0700584 struct htb_sched *q = qdisc_priv(sch);
585 struct htb_class *cl = htb_classify(skb, sch, &ret);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700586
Stephen Hemminger87990462006-08-10 23:35:16 -0700587 if (cl == HTB_DIRECT) {
588 /* enqueue to helper queue */
589 if (q->direct_queue.qlen < q->direct_qlen) {
David S. Milleraea890b2018-07-29 16:22:13 -0700590 __qdisc_enqueue_tail(skb, &q->direct_queue);
Stephen Hemminger87990462006-08-10 23:35:16 -0700591 q->direct_pkts++;
592 } else {
Eric Dumazet520ac302016-06-21 23:16:49 -0700593 return qdisc_drop(skb, sch, to_free);
Stephen Hemminger87990462006-08-10 23:35:16 -0700594 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700595#ifdef CONFIG_NET_CLS_ACT
Stephen Hemminger87990462006-08-10 23:35:16 -0700596 } else if (!cl) {
Jarek Poplawskic27f3392008-08-04 22:39:11 -0700597 if (ret & __NET_XMIT_BYPASS)
John Fastabend25331d62014-09-28 11:53:29 -0700598 qdisc_qstats_drop(sch);
Eric Dumazet520ac302016-06-21 23:16:49 -0700599 __qdisc_drop(skb, to_free);
Stephen Hemminger87990462006-08-10 23:35:16 -0700600 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700601#endif
Cong Wang11957be2018-09-07 13:29:14 -0700602 } else if ((ret = qdisc_enqueue(skb, cl->leaf.q,
Eric Dumazet520ac302016-06-21 23:16:49 -0700603 to_free)) != NET_XMIT_SUCCESS) {
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700604 if (net_xmit_drop_count(ret)) {
John Fastabend25331d62014-09-28 11:53:29 -0700605 qdisc_qstats_drop(sch);
Eric Dumazet338ed9b2016-06-21 23:16:51 -0700606 cl->drops++;
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700607 }
David S. Miller69747652008-08-17 23:55:36 -0700608 return ret;
Stephen Hemminger87990462006-08-10 23:35:16 -0700609 } else {
Stephen Hemminger87990462006-08-10 23:35:16 -0700610 htb_activate(q, cl);
611 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700612
Toke Høiland-Jørgensenf6bab192019-01-09 17:09:42 +0100613 sch->qstats.backlog += len;
Stephen Hemminger87990462006-08-10 23:35:16 -0700614 sch->q.qlen++;
Stephen Hemminger87990462006-08-10 23:35:16 -0700615 return NET_XMIT_SUCCESS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700616}
617
Vimalkumar56b765b2012-10-31 06:04:11 +0000618static inline void htb_accnt_tokens(struct htb_class *cl, int bytes, s64 diff)
Jarek Poplawski59e42202008-12-03 21:17:27 -0800619{
Vimalkumar56b765b2012-10-31 06:04:11 +0000620 s64 toks = diff + cl->tokens;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800621
622 if (toks > cl->buffer)
623 toks = cl->buffer;
Jiri Pirko292f1c72013-02-12 00:12:03 +0000624 toks -= (s64) psched_l2t_ns(&cl->rate, bytes);
Jarek Poplawski59e42202008-12-03 21:17:27 -0800625 if (toks <= -cl->mbuffer)
626 toks = 1 - cl->mbuffer;
627
628 cl->tokens = toks;
629}
630
Vimalkumar56b765b2012-10-31 06:04:11 +0000631static inline void htb_accnt_ctokens(struct htb_class *cl, int bytes, s64 diff)
Jarek Poplawski59e42202008-12-03 21:17:27 -0800632{
Vimalkumar56b765b2012-10-31 06:04:11 +0000633 s64 toks = diff + cl->ctokens;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800634
635 if (toks > cl->cbuffer)
636 toks = cl->cbuffer;
Jiri Pirko292f1c72013-02-12 00:12:03 +0000637 toks -= (s64) psched_l2t_ns(&cl->ceil, bytes);
Jarek Poplawski59e42202008-12-03 21:17:27 -0800638 if (toks <= -cl->mbuffer)
639 toks = 1 - cl->mbuffer;
640
641 cl->ctokens = toks;
642}
643
Linus Torvalds1da177e2005-04-16 15:20:36 -0700644/**
645 * htb_charge_class - charges amount "bytes" to leaf and ancestors
646 *
647 * Routine assumes that packet "bytes" long was dequeued from leaf cl
648 * borrowing from "level". It accounts bytes to ceil leaky bucket for
649 * leaf and all ancestors and to rate bucket for ancestors at levels
650 * "level" and higher. It also handles possible change of mode resulting
651 * from the update. Note that mode can also increase here (MAY_BORROW to
652 * CAN_SEND) because we can use more precise clock that event queue here.
653 * In such case we remove class from event queue first.
654 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700655static void htb_charge_class(struct htb_sched *q, struct htb_class *cl,
Ranjit Manomohanc9726d62007-07-10 22:43:16 -0700656 int level, struct sk_buff *skb)
Stephen Hemminger87990462006-08-10 23:35:16 -0700657{
Jussi Kivilinna0abf77e2008-07-20 00:08:27 -0700658 int bytes = qdisc_pkt_len(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700659 enum htb_cmode old_mode;
Vimalkumar56b765b2012-10-31 06:04:11 +0000660 s64 diff;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700661
662 while (cl) {
Vimalkumar56b765b2012-10-31 06:04:11 +0000663 diff = min_t(s64, q->now - cl->t_c, cl->mbuffer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700664 if (cl->level >= level) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700665 if (cl->level == level)
666 cl->xstats.lends++;
Jarek Poplawski59e42202008-12-03 21:17:27 -0800667 htb_accnt_tokens(cl, bytes, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700668 } else {
669 cl->xstats.borrows++;
Stephen Hemminger87990462006-08-10 23:35:16 -0700670 cl->tokens += diff; /* we moved t_c; update tokens */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700671 }
Jarek Poplawski59e42202008-12-03 21:17:27 -0800672 htb_accnt_ctokens(cl, bytes, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700673 cl->t_c = q->now;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700674
Stephen Hemminger87990462006-08-10 23:35:16 -0700675 old_mode = cl->cmode;
676 diff = 0;
677 htb_change_class_mode(q, cl, &diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700678 if (old_mode != cl->cmode) {
679 if (old_mode != HTB_CAN_SEND)
Eric Dumazetc9364632013-06-15 03:30:10 -0700680 htb_safe_rb_erase(&cl->pq_node, &q->hlevel[cl->level].wait_pq);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700681 if (cl->cmode != HTB_CAN_SEND)
Stephen Hemminger87990462006-08-10 23:35:16 -0700682 htb_add_to_wait_tree(q, cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700683 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700684
Eric Dumazetbfe0d022011-01-09 08:30:54 +0000685 /* update basic stats except for leaves which are already updated */
686 if (cl->level)
687 bstats_update(&cl->bstats, skb);
688
Linus Torvalds1da177e2005-04-16 15:20:36 -0700689 cl = cl->parent;
690 }
691}
692
693/**
694 * htb_do_events - make mode changes to classes at the level
695 *
Patrick McHardyfb983d42007-03-16 01:22:39 -0700696 * Scans event queue for pending events and applies them. Returns time of
Jarek Poplawski12247362009-02-01 01:13:22 -0800697 * next pending event (0 for no event in pq, q->now for too many events).
Patrick McHardyfb983d42007-03-16 01:22:39 -0700698 * Note: Applied are events whose have cl->pq_key <= q->now.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700699 */
Eric Dumazetc9364632013-06-15 03:30:10 -0700700static s64 htb_do_events(struct htb_sched *q, const int level,
Eric Dumazet5343a7f2013-06-04 07:11:48 +0000701 unsigned long start)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700702{
Martin Devera8f3ea332008-03-23 22:00:38 -0700703 /* don't run for longer than 2 jiffies; 2 is used instead of
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000704 * 1 to simplify things when jiffy is going to be incremented
705 * too soon
706 */
Jarek Poplawskia73be042009-01-12 21:54:40 -0800707 unsigned long stop_at = start + 2;
Eric Dumazetc9364632013-06-15 03:30:10 -0700708 struct rb_root *wait_pq = &q->hlevel[level].wait_pq;
709
Martin Devera8f3ea332008-03-23 22:00:38 -0700710 while (time_before(jiffies, stop_at)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700711 struct htb_class *cl;
Vimalkumar56b765b2012-10-31 06:04:11 +0000712 s64 diff;
Eric Dumazetc9364632013-06-15 03:30:10 -0700713 struct rb_node *p = rb_first(wait_pq);
Akinbou Mita30bdbe32006-10-12 01:52:05 -0700714
Stephen Hemminger87990462006-08-10 23:35:16 -0700715 if (!p)
716 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700717
718 cl = rb_entry(p, struct htb_class, pq_node);
Patrick McHardyfb983d42007-03-16 01:22:39 -0700719 if (cl->pq_key > q->now)
720 return cl->pq_key;
721
Eric Dumazetc9364632013-06-15 03:30:10 -0700722 htb_safe_rb_erase(p, wait_pq);
Vimalkumar56b765b2012-10-31 06:04:11 +0000723 diff = min_t(s64, q->now - cl->t_c, cl->mbuffer);
Stephen Hemminger87990462006-08-10 23:35:16 -0700724 htb_change_class_mode(q, cl, &diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700725 if (cl->cmode != HTB_CAN_SEND)
Stephen Hemminger87990462006-08-10 23:35:16 -0700726 htb_add_to_wait_tree(q, cl, diff);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700727 }
Jarek Poplawski12247362009-02-01 01:13:22 -0800728
729 /* too much load - let's continue after a break for scheduling */
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800730 if (!(q->warned & HTB_WARN_TOOMANYEVENTS)) {
Yang Yingliangc17988a2013-12-23 17:38:58 +0800731 pr_warn("htb: too many events!\n");
Jarek Poplawskie82181d2009-02-01 01:13:05 -0800732 q->warned |= HTB_WARN_TOOMANYEVENTS;
733 }
Jarek Poplawski12247362009-02-01 01:13:22 -0800734
735 return q->now;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700736}
737
738/* Returns class->node+prio from id-tree where classe's id is >= id. NULL
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000739 * is no such one exists.
740 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700741static struct rb_node *htb_id_find_next_upper(int prio, struct rb_node *n,
742 u32 id)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700743{
744 struct rb_node *r = NULL;
745 while (n) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700746 struct htb_class *cl =
747 rb_entry(n, struct htb_class, node[prio]);
Stephen Hemminger87990462006-08-10 23:35:16 -0700748
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700749 if (id > cl->common.classid) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700750 n = n->rb_right;
Jarek Poplawski1b5c0072008-12-09 22:34:40 -0800751 } else if (id < cl->common.classid) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700752 r = n;
753 n = n->rb_left;
Jarek Poplawski1b5c0072008-12-09 22:34:40 -0800754 } else {
755 return n;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756 }
757 }
758 return r;
759}
760
761/**
762 * htb_lookup_leaf - returns next leaf class in DRR order
763 *
764 * Find leaf where current feed pointers points to.
765 */
Eric Dumazetc9364632013-06-15 03:30:10 -0700766static struct htb_class *htb_lookup_leaf(struct htb_prio *hprio, const int prio)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700767{
768 int i;
769 struct {
770 struct rb_node *root;
771 struct rb_node **pptr;
772 u32 *pid;
Stephen Hemminger87990462006-08-10 23:35:16 -0700773 } stk[TC_HTB_MAXDEPTH], *sp = stk;
774
Eric Dumazetc9364632013-06-15 03:30:10 -0700775 BUG_ON(!hprio->row.rb_node);
776 sp->root = hprio->row.rb_node;
777 sp->pptr = &hprio->ptr;
778 sp->pid = &hprio->last_ptr_id;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700779
780 for (i = 0; i < 65535; i++) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700781 if (!*sp->pptr && *sp->pid) {
YOSHIFUJI Hideaki10297b92007-02-09 23:25:16 +0900782 /* ptr was invalidated but id is valid - try to recover
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000783 * the original or next ptr
784 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700785 *sp->pptr =
786 htb_id_find_next_upper(prio, sp->root, *sp->pid);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700787 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700788 *sp->pid = 0; /* ptr is valid now so that remove this hint as it
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000789 * can become out of date quickly
790 */
Stephen Hemminger87990462006-08-10 23:35:16 -0700791 if (!*sp->pptr) { /* we are at right end; rewind & go up */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700792 *sp->pptr = sp->root;
Stephen Hemminger87990462006-08-10 23:35:16 -0700793 while ((*sp->pptr)->rb_left)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700794 *sp->pptr = (*sp->pptr)->rb_left;
795 if (sp > stk) {
796 sp--;
Jarek Poplawski512bb432008-12-09 22:35:02 -0800797 if (!*sp->pptr) {
798 WARN_ON(1);
Stephen Hemminger87990462006-08-10 23:35:16 -0700799 return NULL;
Jarek Poplawski512bb432008-12-09 22:35:02 -0800800 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700801 htb_next_rb_node(sp->pptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700802 }
803 } else {
804 struct htb_class *cl;
Eric Dumazetc9364632013-06-15 03:30:10 -0700805 struct htb_prio *clp;
806
Stephen Hemminger87990462006-08-10 23:35:16 -0700807 cl = rb_entry(*sp->pptr, struct htb_class, node[prio]);
808 if (!cl->level)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700809 return cl;
Cong Wang11957be2018-09-07 13:29:14 -0700810 clp = &cl->inner.clprio[prio];
Eric Dumazetc9364632013-06-15 03:30:10 -0700811 (++sp)->root = clp->feed.rb_node;
812 sp->pptr = &clp->ptr;
813 sp->pid = &clp->last_ptr_id;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700814 }
815 }
Ilpo Järvinen547b7922008-07-25 21:43:18 -0700816 WARN_ON(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700817 return NULL;
818}
819
820/* dequeues packet at given priority and level; call only if
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000821 * you are sure that there is active class at prio/level
822 */
Eric Dumazetc9364632013-06-15 03:30:10 -0700823static struct sk_buff *htb_dequeue_tree(struct htb_sched *q, const int prio,
824 const int level)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700825{
826 struct sk_buff *skb = NULL;
Stephen Hemminger87990462006-08-10 23:35:16 -0700827 struct htb_class *cl, *start;
Eric Dumazetc9364632013-06-15 03:30:10 -0700828 struct htb_level *hlevel = &q->hlevel[level];
829 struct htb_prio *hprio = &hlevel->hprio[prio];
830
Linus Torvalds1da177e2005-04-16 15:20:36 -0700831 /* look initial class up in the row */
Eric Dumazetc9364632013-06-15 03:30:10 -0700832 start = cl = htb_lookup_leaf(hprio, prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700833
Linus Torvalds1da177e2005-04-16 15:20:36 -0700834 do {
835next:
Jarek Poplawski512bb432008-12-09 22:35:02 -0800836 if (unlikely(!cl))
Stephen Hemminger87990462006-08-10 23:35:16 -0700837 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700838
839 /* class can be empty - it is unlikely but can be true if leaf
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000840 * qdisc drops packets in enqueue routine or if someone used
841 * graft operation on the leaf since last dequeue;
842 * simply deactivate and skip such class
843 */
Cong Wang11957be2018-09-07 13:29:14 -0700844 if (unlikely(cl->leaf.q->q.qlen == 0)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700845 struct htb_class *next;
Stephen Hemminger87990462006-08-10 23:35:16 -0700846 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700847
848 /* row/level might become empty */
849 if ((q->row_mask[level] & (1 << prio)) == 0)
Stephen Hemminger87990462006-08-10 23:35:16 -0700850 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700851
Eric Dumazetc9364632013-06-15 03:30:10 -0700852 next = htb_lookup_leaf(hprio, prio);
Stephen Hemminger87990462006-08-10 23:35:16 -0700853
854 if (cl == start) /* fix start if we just deleted it */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700855 start = next;
856 cl = next;
857 goto next;
858 }
Stephen Hemminger87990462006-08-10 23:35:16 -0700859
Cong Wang11957be2018-09-07 13:29:14 -0700860 skb = cl->leaf.q->dequeue(cl->leaf.q);
Stephen Hemminger87990462006-08-10 23:35:16 -0700861 if (likely(skb != NULL))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700862 break;
Jarek Poplawski633fe662008-12-03 21:09:10 -0800863
Cong Wang11957be2018-09-07 13:29:14 -0700864 qdisc_warn_nonwc("htb", cl->leaf.q);
865 htb_next_rb_node(level ? &cl->parent->inner.clprio[prio].ptr:
Eric Dumazetc9364632013-06-15 03:30:10 -0700866 &q->hlevel[0].hprio[prio].ptr);
867 cl = htb_lookup_leaf(hprio, prio);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700868
869 } while (cl != start);
870
871 if (likely(skb != NULL)) {
Eric Dumazet196d97f2012-11-05 16:40:49 +0000872 bstats_update(&cl->bstats, skb);
Cong Wang11957be2018-09-07 13:29:14 -0700873 cl->leaf.deficit[level] -= qdisc_pkt_len(skb);
874 if (cl->leaf.deficit[level] < 0) {
875 cl->leaf.deficit[level] += cl->quantum;
876 htb_next_rb_node(level ? &cl->parent->inner.clprio[prio].ptr :
Eric Dumazetc9364632013-06-15 03:30:10 -0700877 &q->hlevel[0].hprio[prio].ptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700878 }
879 /* this used to be after charge_class but this constelation
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000880 * gives us slightly better performance
881 */
Cong Wang11957be2018-09-07 13:29:14 -0700882 if (!cl->leaf.q->q.qlen)
Stephen Hemminger87990462006-08-10 23:35:16 -0700883 htb_deactivate(q, cl);
Ranjit Manomohanc9726d62007-07-10 22:43:16 -0700884 htb_charge_class(q, cl, level, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700885 }
886 return skb;
887}
888
Linus Torvalds1da177e2005-04-16 15:20:36 -0700889static struct sk_buff *htb_dequeue(struct Qdisc *sch)
890{
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800891 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700892 struct htb_sched *q = qdisc_priv(sch);
893 int level;
Eric Dumazet5343a7f2013-06-04 07:11:48 +0000894 s64 next_event;
Jarek Poplawskia73be042009-01-12 21:54:40 -0800895 unsigned long start_at;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700896
897 /* try to dequeue direct packets as high prio (!) to minimize cpu work */
Florian Westphal48da34b2016-09-18 00:57:34 +0200898 skb = __qdisc_dequeue_head(&q->direct_queue);
Stephen Hemminger87990462006-08-10 23:35:16 -0700899 if (skb != NULL) {
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800900ok:
901 qdisc_bstats_update(sch, skb);
WANG Cong431e3a82016-02-25 14:55:02 -0800902 qdisc_qstats_backlog_dec(sch, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700903 sch->q.qlen--;
904 return skb;
905 }
906
Stephen Hemminger87990462006-08-10 23:35:16 -0700907 if (!sch->q.qlen)
908 goto fin;
Eric Dumazetd2de8752014-08-22 18:32:09 -0700909 q->now = ktime_get_ns();
Jarek Poplawskia73be042009-01-12 21:54:40 -0800910 start_at = jiffies;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700911
Stefan Haskod2fe85d2012-12-21 15:04:59 +0000912 next_event = q->now + 5LLU * NSEC_PER_SEC;
Jarek Poplawski633fe662008-12-03 21:09:10 -0800913
Linus Torvalds1da177e2005-04-16 15:20:36 -0700914 for (level = 0; level < TC_HTB_MAXDEPTH; level++) {
915 /* common case optimization - skip event handler quickly */
916 int m;
Eric Dumazetc9364632013-06-15 03:30:10 -0700917 s64 event = q->near_ev_cache[level];
Stephen Hemminger87990462006-08-10 23:35:16 -0700918
Eric Dumazetc9364632013-06-15 03:30:10 -0700919 if (q->now >= event) {
Jarek Poplawskia73be042009-01-12 21:54:40 -0800920 event = htb_do_events(q, level, start_at);
Patrick McHardy2e4b3b02007-05-23 23:39:54 -0700921 if (!event)
Vimalkumar56b765b2012-10-31 06:04:11 +0000922 event = q->now + NSEC_PER_SEC;
Patrick McHardy2e4b3b02007-05-23 23:39:54 -0700923 q->near_ev_cache[level] = event;
Eric Dumazetc9364632013-06-15 03:30:10 -0700924 }
Patrick McHardyfb983d42007-03-16 01:22:39 -0700925
Jarek Poplawskic0851342009-01-12 21:54:16 -0800926 if (next_event > event)
Patrick McHardyfb983d42007-03-16 01:22:39 -0700927 next_event = event;
928
Linus Torvalds1da177e2005-04-16 15:20:36 -0700929 m = ~q->row_mask[level];
930 while (m != (int)(-1)) {
Stephen Hemminger87990462006-08-10 23:35:16 -0700931 int prio = ffz(m);
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000932
Linus Torvalds1da177e2005-04-16 15:20:36 -0700933 m |= 1 << prio;
Stephen Hemminger87990462006-08-10 23:35:16 -0700934 skb = htb_dequeue_tree(q, prio, level);
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800935 if (likely(skb != NULL))
936 goto ok;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700937 }
938 }
Eric Dumazeta9efad82016-05-23 14:24:56 -0700939 if (likely(next_event > q->now))
Eric Dumazet45f50be2016-06-10 16:41:39 -0700940 qdisc_watchdog_schedule_ns(&q->watchdog, next_event);
Eric Dumazeta9efad82016-05-23 14:24:56 -0700941 else
Jarek Poplawski12247362009-02-01 01:13:22 -0800942 schedule_work(&q->work);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700943fin:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700944 return skb;
945}
946
Linus Torvalds1da177e2005-04-16 15:20:36 -0700947/* reset all classes */
948/* always caled under BH & queue lock */
Stephen Hemminger87990462006-08-10 23:35:16 -0700949static void htb_reset(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700950{
951 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700952 struct htb_class *cl;
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700953 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700954
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -0700955 for (i = 0; i < q->clhash.hashsize; i++) {
Sasha Levinb67bfe02013-02-27 17:06:00 -0800956 hlist_for_each_entry(cl, &q->clhash.hash[i], common.hnode) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700957 if (cl->level)
Cong Wang11957be2018-09-07 13:29:14 -0700958 memset(&cl->inner, 0, sizeof(cl->inner));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700959 else {
Cong Wang11957be2018-09-07 13:29:14 -0700960 if (cl->leaf.q)
961 qdisc_reset(cl->leaf.q);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700962 }
963 cl->prio_activity = 0;
964 cl->cmode = HTB_CAN_SEND;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700965 }
966 }
Patrick McHardyfb983d42007-03-16 01:22:39 -0700967 qdisc_watchdog_cancel(&q->watchdog);
Eric Dumazeta5a9f5342016-06-13 20:21:56 -0700968 __qdisc_reset_queue(&q->direct_queue);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700969 sch->q.qlen = 0;
WANG Cong431e3a82016-02-25 14:55:02 -0800970 sch->qstats.backlog = 0;
Eric Dumazetc9364632013-06-15 03:30:10 -0700971 memset(q->hlevel, 0, sizeof(q->hlevel));
Stephen Hemminger87990462006-08-10 23:35:16 -0700972 memset(q->row_mask, 0, sizeof(q->row_mask));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700973}
974
Patrick McHardy27a34212008-01-23 20:35:39 -0800975static const struct nla_policy htb_policy[TCA_HTB_MAX + 1] = {
976 [TCA_HTB_PARMS] = { .len = sizeof(struct tc_htb_opt) },
977 [TCA_HTB_INIT] = { .len = sizeof(struct tc_htb_glob) },
978 [TCA_HTB_CTAB] = { .type = NLA_BINARY, .len = TC_RTAB_SIZE },
979 [TCA_HTB_RTAB] = { .type = NLA_BINARY, .len = TC_RTAB_SIZE },
Eric Dumazet6906f4e2013-03-06 06:49:21 +0000980 [TCA_HTB_DIRECT_QLEN] = { .type = NLA_U32 },
Eric Dumazetdf62cdf2013-09-19 09:10:20 -0700981 [TCA_HTB_RATE64] = { .type = NLA_U64 },
982 [TCA_HTB_CEIL64] = { .type = NLA_U64 },
Patrick McHardy27a34212008-01-23 20:35:39 -0800983};
984
Jarek Poplawski12247362009-02-01 01:13:22 -0800985static void htb_work_func(struct work_struct *work)
986{
987 struct htb_sched *q = container_of(work, struct htb_sched, work);
988 struct Qdisc *sch = q->watchdog.qdisc;
989
Florian Westphal0ee13622016-06-14 06:16:27 +0200990 rcu_read_lock();
Jarek Poplawski12247362009-02-01 01:13:22 -0800991 __netif_schedule(qdisc_root(sch));
Florian Westphal0ee13622016-06-14 06:16:27 +0200992 rcu_read_unlock();
Jarek Poplawski12247362009-02-01 01:13:22 -0800993}
994
Alexander Aringe63d7df2017-12-20 12:35:13 -0500995static int htb_init(struct Qdisc *sch, struct nlattr *opt,
996 struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700997{
998 struct htb_sched *q = qdisc_priv(sch);
Eric Dumazet6906f4e2013-03-06 06:49:21 +0000999 struct nlattr *tb[TCA_HTB_MAX + 1];
Linus Torvalds1da177e2005-04-16 15:20:36 -07001000 struct tc_htb_glob *gopt;
Patrick McHardycee63722008-01-23 20:33:32 -08001001 int err;
Patrick McHardycee63722008-01-23 20:33:32 -08001002
Nikolay Aleksandrov88c2ace2017-08-30 12:48:57 +03001003 qdisc_watchdog_init(&q->watchdog, sch);
1004 INIT_WORK(&q->work, htb_work_func);
1005
Patrick McHardycee63722008-01-23 20:33:32 -08001006 if (!opt)
1007 return -EINVAL;
1008
Alexander Aring8d1a77f2017-12-20 12:35:19 -05001009 err = tcf_block_get(&q->block, &q->filter_list, sch, extack);
Jiri Pirko6529eab2017-05-17 11:07:55 +02001010 if (err)
1011 return err;
1012
Johannes Berg8cb08172019-04-26 14:07:28 +02001013 err = nla_parse_nested_deprecated(tb, TCA_HTB_MAX, opt, htb_policy,
1014 NULL);
Patrick McHardycee63722008-01-23 20:33:32 -08001015 if (err < 0)
1016 return err;
1017
Eric Dumazet6906f4e2013-03-06 06:49:21 +00001018 if (!tb[TCA_HTB_INIT])
Linus Torvalds1da177e2005-04-16 15:20:36 -07001019 return -EINVAL;
Eric Dumazet6906f4e2013-03-06 06:49:21 +00001020
Patrick McHardy1e904742008-01-22 22:11:17 -08001021 gopt = nla_data(tb[TCA_HTB_INIT]);
Eric Dumazet6906f4e2013-03-06 06:49:21 +00001022 if (gopt->version != HTB_VER >> 16)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001023 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001024
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001025 err = qdisc_class_hash_init(&q->clhash);
1026 if (err < 0)
1027 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001028
Florian Westphal48da34b2016-09-18 00:57:34 +02001029 qdisc_skb_head_init(&q->direct_queue);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001030
Eric Dumazet6906f4e2013-03-06 06:49:21 +00001031 if (tb[TCA_HTB_DIRECT_QLEN])
1032 q->direct_qlen = nla_get_u32(tb[TCA_HTB_DIRECT_QLEN]);
Phil Sutter348e3432015-08-18 10:30:49 +02001033 else
Eric Dumazet6906f4e2013-03-06 06:49:21 +00001034 q->direct_qlen = qdisc_dev(sch)->tx_queue_len;
Phil Sutter348e3432015-08-18 10:30:49 +02001035
Linus Torvalds1da177e2005-04-16 15:20:36 -07001036 if ((q->rate2quantum = gopt->rate2quantum) < 1)
1037 q->rate2quantum = 1;
1038 q->defcls = gopt->defcls;
1039
1040 return 0;
1041}
1042
1043static int htb_dump(struct Qdisc *sch, struct sk_buff *skb)
1044{
1045 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001046 struct nlattr *nest;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001047 struct tc_htb_glob gopt;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001048
Cong Wangb3624872019-05-04 11:43:42 -07001049 sch->qstats.overlimits = q->overlimits;
Eric Dumazet6f542ef2014-03-05 10:14:34 -08001050 /* Its safe to not acquire qdisc lock. As we hold RTNL,
1051 * no change can happen on the qdisc parameters.
1052 */
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001053
1054 gopt.direct_pkts = q->direct_pkts;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001055 gopt.version = HTB_VER;
1056 gopt.rate2quantum = q->rate2quantum;
1057 gopt.defcls = q->defcls;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001058 gopt.debug = 0;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001059
Michal Kubecekae0be8d2019-04-26 11:13:06 +02001060 nest = nla_nest_start_noflag(skb, TCA_OPTIONS);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001061 if (nest == NULL)
1062 goto nla_put_failure;
Eric Dumazet6906f4e2013-03-06 06:49:21 +00001063 if (nla_put(skb, TCA_HTB_INIT, sizeof(gopt), &gopt) ||
1064 nla_put_u32(skb, TCA_HTB_DIRECT_QLEN, q->direct_qlen))
David S. Miller1b34ec42012-03-29 05:11:39 -04001065 goto nla_put_failure;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001066
Eric Dumazet6f542ef2014-03-05 10:14:34 -08001067 return nla_nest_end(skb, nest);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001068
Patrick McHardy1e904742008-01-22 22:11:17 -08001069nla_put_failure:
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001070 nla_nest_cancel(skb, nest);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001071 return -1;
1072}
1073
1074static int htb_dump_class(struct Qdisc *sch, unsigned long arg,
Stephen Hemminger87990462006-08-10 23:35:16 -07001075 struct sk_buff *skb, struct tcmsg *tcm)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001076{
Stephen Hemminger87990462006-08-10 23:35:16 -07001077 struct htb_class *cl = (struct htb_class *)arg;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001078 struct nlattr *nest;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001079 struct tc_htb_opt opt;
1080
Eric Dumazet6f542ef2014-03-05 10:14:34 -08001081 /* Its safe to not acquire qdisc lock. As we hold RTNL,
1082 * no change can happen on the class parameters.
1083 */
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001084 tcm->tcm_parent = cl->parent ? cl->parent->common.classid : TC_H_ROOT;
1085 tcm->tcm_handle = cl->common.classid;
Cong Wang11957be2018-09-07 13:29:14 -07001086 if (!cl->level && cl->leaf.q)
1087 tcm->tcm_info = cl->leaf.q->handle;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001088
Michal Kubecekae0be8d2019-04-26 11:13:06 +02001089 nest = nla_nest_start_noflag(skb, TCA_OPTIONS);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001090 if (nest == NULL)
1091 goto nla_put_failure;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001092
Stephen Hemminger87990462006-08-10 23:35:16 -07001093 memset(&opt, 0, sizeof(opt));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001094
Eric Dumazet01cb71d2013-06-02 13:55:05 +00001095 psched_ratecfg_getrate(&opt.rate, &cl->rate);
Jiri Pirko9c10f412013-02-12 00:12:00 +00001096 opt.buffer = PSCHED_NS2TICKS(cl->buffer);
Eric Dumazet01cb71d2013-06-02 13:55:05 +00001097 psched_ratecfg_getrate(&opt.ceil, &cl->ceil);
Jiri Pirko9c10f412013-02-12 00:12:00 +00001098 opt.cbuffer = PSCHED_NS2TICKS(cl->cbuffer);
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001099 opt.quantum = cl->quantum;
1100 opt.prio = cl->prio;
Stephen Hemminger87990462006-08-10 23:35:16 -07001101 opt.level = cl->level;
David S. Miller1b34ec42012-03-29 05:11:39 -04001102 if (nla_put(skb, TCA_HTB_PARMS, sizeof(opt), &opt))
1103 goto nla_put_failure;
Eric Dumazetdf62cdf2013-09-19 09:10:20 -07001104 if ((cl->rate.rate_bytes_ps >= (1ULL << 32)) &&
Nicolas Dichtel2a51c1e2016-04-25 10:25:15 +02001105 nla_put_u64_64bit(skb, TCA_HTB_RATE64, cl->rate.rate_bytes_ps,
1106 TCA_HTB_PAD))
Eric Dumazetdf62cdf2013-09-19 09:10:20 -07001107 goto nla_put_failure;
1108 if ((cl->ceil.rate_bytes_ps >= (1ULL << 32)) &&
Nicolas Dichtel2a51c1e2016-04-25 10:25:15 +02001109 nla_put_u64_64bit(skb, TCA_HTB_CEIL64, cl->ceil.rate_bytes_ps,
1110 TCA_HTB_PAD))
Eric Dumazetdf62cdf2013-09-19 09:10:20 -07001111 goto nla_put_failure;
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001112
Eric Dumazet6f542ef2014-03-05 10:14:34 -08001113 return nla_nest_end(skb, nest);
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001114
Patrick McHardy1e904742008-01-22 22:11:17 -08001115nla_put_failure:
Patrick McHardy4b3550ef2008-01-23 20:34:11 -08001116 nla_nest_cancel(skb, nest);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001117 return -1;
1118}
1119
1120static int
Stephen Hemminger87990462006-08-10 23:35:16 -07001121htb_dump_class_stats(struct Qdisc *sch, unsigned long arg, struct gnet_dump *d)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001122{
Stephen Hemminger87990462006-08-10 23:35:16 -07001123 struct htb_class *cl = (struct htb_class *)arg;
Eric Dumazet338ed9b2016-06-21 23:16:51 -07001124 struct gnet_stats_queue qs = {
1125 .drops = cl->drops,
Eric Dumazet3c75f6e2017-09-18 12:36:22 -07001126 .overlimits = cl->overlimits,
Eric Dumazet338ed9b2016-06-21 23:16:51 -07001127 };
John Fastabend64015852014-09-28 11:53:57 -07001128 __u32 qlen = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001129
Paolo Abeni5dd431b2019-03-28 16:53:12 +01001130 if (!cl->level && cl->leaf.q)
1131 qdisc_qstats_qlen_backlog(cl->leaf.q, &qlen, &qs.backlog);
1132
Konstantin Khlebnikov0564bf02016-07-16 17:08:56 +03001133 cl->xstats.tokens = clamp_t(s64, PSCHED_NS2TICKS(cl->tokens),
1134 INT_MIN, INT_MAX);
1135 cl->xstats.ctokens = clamp_t(s64, PSCHED_NS2TICKS(cl->ctokens),
1136 INT_MIN, INT_MAX);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001137
Eric Dumazetedb09eb2016-06-06 09:37:16 -07001138 if (gnet_stats_copy_basic(qdisc_root_sleeping_running(sch),
1139 d, NULL, &cl->bstats) < 0 ||
Eric Dumazet1c0d32f2016-12-04 09:48:16 -08001140 gnet_stats_copy_rate_est(d, &cl->rate_est) < 0 ||
Eric Dumazet338ed9b2016-06-21 23:16:51 -07001141 gnet_stats_copy_queue(d, NULL, &qs, qlen) < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001142 return -1;
1143
1144 return gnet_stats_copy_app(d, &cl->xstats, sizeof(cl->xstats));
1145}
1146
1147static int htb_graft(struct Qdisc *sch, unsigned long arg, struct Qdisc *new,
Alexander Aring653d6fd2017-12-20 12:35:17 -05001148 struct Qdisc **old, struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001149{
Stephen Hemminger87990462006-08-10 23:35:16 -07001150 struct htb_class *cl = (struct htb_class *)arg;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001151
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001152 if (cl->level)
1153 return -EINVAL;
1154 if (new == NULL &&
Changli Gao3511c912010-10-16 13:04:08 +00001155 (new = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops,
Alexander Aringa38a9882017-12-20 12:35:21 -05001156 cl->common.classid, extack)) == NULL)
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001157 return -ENOBUFS;
1158
Cong Wang11957be2018-09-07 13:29:14 -07001159 *old = qdisc_replace(sch, new, &cl->leaf.q);
Patrick McHardy5b9a9cc2009-09-04 06:41:17 +00001160 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161}
1162
Stephen Hemminger87990462006-08-10 23:35:16 -07001163static struct Qdisc *htb_leaf(struct Qdisc *sch, unsigned long arg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001164{
Stephen Hemminger87990462006-08-10 23:35:16 -07001165 struct htb_class *cl = (struct htb_class *)arg;
Cong Wang11957be2018-09-07 13:29:14 -07001166 return !cl->level ? cl->leaf.q : NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001167}
1168
Patrick McHardy256d61b2006-11-29 17:37:05 -08001169static void htb_qlen_notify(struct Qdisc *sch, unsigned long arg)
1170{
1171 struct htb_class *cl = (struct htb_class *)arg;
1172
Konstantin Khlebnikov95946652017-08-15 16:39:59 +03001173 htb_deactivate(qdisc_priv(sch), cl);
Patrick McHardy256d61b2006-11-29 17:37:05 -08001174}
1175
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001176static inline int htb_parent_last_child(struct htb_class *cl)
1177{
1178 if (!cl->parent)
1179 /* the root class */
1180 return 0;
Patrick McHardy42077592008-07-05 23:22:53 -07001181 if (cl->parent->children > 1)
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001182 /* not the last child */
1183 return 0;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001184 return 1;
1185}
1186
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001187static void htb_parent_to_leaf(struct htb_sched *q, struct htb_class *cl,
1188 struct Qdisc *new_q)
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001189{
1190 struct htb_class *parent = cl->parent;
1191
Cong Wang11957be2018-09-07 13:29:14 -07001192 WARN_ON(cl->level || !cl->leaf.q || cl->prio_activity);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001193
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001194 if (parent->cmode != HTB_CAN_SEND)
Eric Dumazetc9364632013-06-15 03:30:10 -07001195 htb_safe_rb_erase(&parent->pq_node,
1196 &q->hlevel[parent->level].wait_pq);
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001197
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001198 parent->level = 0;
Cong Wang11957be2018-09-07 13:29:14 -07001199 memset(&parent->inner, 0, sizeof(parent->inner));
1200 parent->leaf.q = new_q ? new_q : &noop_qdisc;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001201 parent->tokens = parent->buffer;
1202 parent->ctokens = parent->cbuffer;
Eric Dumazetd2de8752014-08-22 18:32:09 -07001203 parent->t_c = ktime_get_ns();
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001204 parent->cmode = HTB_CAN_SEND;
1205}
1206
Stephen Hemminger87990462006-08-10 23:35:16 -07001207static void htb_destroy_class(struct Qdisc *sch, struct htb_class *cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001208{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001209 if (!cl->level) {
Cong Wang11957be2018-09-07 13:29:14 -07001210 WARN_ON(!cl->leaf.q);
Vlad Buslov86bd4462018-09-24 19:22:50 +03001211 qdisc_put(cl->leaf.q);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001212 }
Eric Dumazet1c0d32f2016-12-04 09:48:16 -08001213 gen_kill_estimator(&cl->rate_est);
Jiri Pirko6529eab2017-05-17 11:07:55 +02001214 tcf_block_put(cl->block);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001215 kfree(cl);
1216}
1217
Stephen Hemminger87990462006-08-10 23:35:16 -07001218static void htb_destroy(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001219{
1220 struct htb_sched *q = qdisc_priv(sch);
Sasha Levinb67bfe02013-02-27 17:06:00 -08001221 struct hlist_node *next;
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001222 struct htb_class *cl;
1223 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001224
Jarek Poplawski12247362009-02-01 01:13:22 -08001225 cancel_work_sync(&q->work);
Patrick McHardyfb983d42007-03-16 01:22:39 -07001226 qdisc_watchdog_cancel(&q->watchdog);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001227 /* This line used to be after htb_destroy_class call below
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001228 * and surprisingly it worked in 2.4. But it must precede it
1229 * because filter need its target class alive to be able to call
1230 * unbind_filter on it (without Oops).
1231 */
Jiri Pirko6529eab2017-05-17 11:07:55 +02001232 tcf_block_put(q->block);
Stephen Hemminger87990462006-08-10 23:35:16 -07001233
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001234 for (i = 0; i < q->clhash.hashsize; i++) {
Konstantin Khlebnikov89890422017-08-15 16:35:21 +03001235 hlist_for_each_entry(cl, &q->clhash.hash[i], common.hnode) {
Jiri Pirko6529eab2017-05-17 11:07:55 +02001236 tcf_block_put(cl->block);
Konstantin Khlebnikov89890422017-08-15 16:35:21 +03001237 cl->block = NULL;
1238 }
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001239 }
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001240 for (i = 0; i < q->clhash.hashsize; i++) {
Sasha Levinb67bfe02013-02-27 17:06:00 -08001241 hlist_for_each_entry_safe(cl, next, &q->clhash.hash[i],
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001242 common.hnode)
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001243 htb_destroy_class(sch, cl);
1244 }
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001245 qdisc_class_hash_destroy(&q->clhash);
Eric Dumazeta5a9f5342016-06-13 20:21:56 -07001246 __qdisc_reset_queue(&q->direct_queue);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001247}
1248
1249static int htb_delete(struct Qdisc *sch, unsigned long arg)
1250{
1251 struct htb_sched *q = qdisc_priv(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001252 struct htb_class *cl = (struct htb_class *)arg;
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001253 struct Qdisc *new_q = NULL;
1254 int last_child = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001255
Yang Yinglianga071d272013-12-23 17:38:59 +08001256 /* TODO: why don't allow to delete subtree ? references ? does
1257 * tc subsys guarantee us that in htb_destroy it holds no class
1258 * refs so that we can remove children safely there ?
1259 */
Patrick McHardy42077592008-07-05 23:22:53 -07001260 if (cl->children || cl->filter_cnt)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001261 return -EBUSY;
Stephen Hemminger87990462006-08-10 23:35:16 -07001262
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001263 if (!cl->level && htb_parent_last_child(cl)) {
Changli Gao3511c912010-10-16 13:04:08 +00001264 new_q = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops,
Alexander Aringa38a9882017-12-20 12:35:21 -05001265 cl->parent->common.classid,
1266 NULL);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001267 last_child = 1;
1268 }
1269
Linus Torvalds1da177e2005-04-16 15:20:36 -07001270 sch_tree_lock(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001271
Paolo Abenie5f0e8f2019-03-28 16:53:13 +01001272 if (!cl->level)
1273 qdisc_purge_queue(cl->leaf.q);
Patrick McHardy814a175e2006-11-29 17:34:50 -08001274
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001275 /* delete from hash and active; remainder in destroy_class */
1276 qdisc_class_hash_remove(&q->clhash, &cl->common);
Jarek Poplawski26b284d2008-08-13 15:16:43 -07001277 if (cl->parent)
1278 cl->parent->children--;
Patrick McHardyc38c83c2007-03-27 14:04:24 -07001279
Linus Torvalds1da177e2005-04-16 15:20:36 -07001280 if (cl->prio_activity)
Stephen Hemminger87990462006-08-10 23:35:16 -07001281 htb_deactivate(q, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001282
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001283 if (cl->cmode != HTB_CAN_SEND)
Eric Dumazetc9364632013-06-15 03:30:10 -07001284 htb_safe_rb_erase(&cl->pq_node,
1285 &q->hlevel[cl->level].wait_pq);
Patrick McHardyfbd8f132008-07-05 23:22:19 -07001286
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001287 if (last_child)
Jarek Poplawski3ba08b02008-05-03 20:46:29 -07001288 htb_parent_to_leaf(q, cl, new_q);
Jarek Poplawski160d5e12006-12-08 00:26:56 -08001289
Linus Torvalds1da177e2005-04-16 15:20:36 -07001290 sch_tree_unlock(sch);
WANG Cong143976c2017-08-24 16:51:29 -07001291
1292 htb_destroy_class(sch, cl);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001293 return 0;
1294}
1295
Stephen Hemminger87990462006-08-10 23:35:16 -07001296static int htb_change_class(struct Qdisc *sch, u32 classid,
Patrick McHardy1e904742008-01-22 22:11:17 -08001297 u32 parentid, struct nlattr **tca,
Alexander Aring793d81d2017-12-20 12:35:15 -05001298 unsigned long *arg, struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001299{
1300 int err = -EINVAL;
1301 struct htb_sched *q = qdisc_priv(sch);
Stephen Hemminger87990462006-08-10 23:35:16 -07001302 struct htb_class *cl = (struct htb_class *)*arg, *parent;
Patrick McHardy1e904742008-01-22 22:11:17 -08001303 struct nlattr *opt = tca[TCA_OPTIONS];
Eric Dumazet6906f4e2013-03-06 06:49:21 +00001304 struct nlattr *tb[TCA_HTB_MAX + 1];
Vlad Buslov4ce70b42019-09-24 18:51:16 +03001305 struct Qdisc *parent_qdisc = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001306 struct tc_htb_opt *hopt;
Eric Dumazetdf62cdf2013-09-19 09:10:20 -07001307 u64 rate64, ceil64;
Li RongQingda01ec42018-03-30 10:11:21 +08001308 int warn = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001309
1310 /* extract all subattrs from opt attr */
Patrick McHardycee63722008-01-23 20:33:32 -08001311 if (!opt)
1312 goto failure;
1313
Johannes Berg8cb08172019-04-26 14:07:28 +02001314 err = nla_parse_nested_deprecated(tb, TCA_HTB_MAX, opt, htb_policy,
1315 NULL);
Patrick McHardycee63722008-01-23 20:33:32 -08001316 if (err < 0)
1317 goto failure;
1318
1319 err = -EINVAL;
Patrick McHardy27a34212008-01-23 20:35:39 -08001320 if (tb[TCA_HTB_PARMS] == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001321 goto failure;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001322
Stephen Hemminger87990462006-08-10 23:35:16 -07001323 parent = parentid == TC_H_ROOT ? NULL : htb_find(parentid, sch);
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001324
Patrick McHardy1e904742008-01-22 22:11:17 -08001325 hopt = nla_data(tb[TCA_HTB_PARMS]);
Eric Dumazet196d97f2012-11-05 16:40:49 +00001326 if (!hopt->rate.rate || !hopt->ceil.rate)
Stephen Hemminger87990462006-08-10 23:35:16 -07001327 goto failure;
1328
Jesper Dangaard Brouer8a8e3d82013-08-14 23:47:11 +02001329 /* Keeping backward compatible with rate_table based iproute2 tc */
Yang Yingliang6b1dd852013-12-11 15:48:37 +08001330 if (hopt->rate.linklayer == TC_LINKLAYER_UNAWARE)
Alexander Aringe9bc3fa2017-12-20 12:35:18 -05001331 qdisc_put_rtab(qdisc_get_rtab(&hopt->rate, tb[TCA_HTB_RTAB],
1332 NULL));
Yang Yingliang6b1dd852013-12-11 15:48:37 +08001333
1334 if (hopt->ceil.linklayer == TC_LINKLAYER_UNAWARE)
Alexander Aringe9bc3fa2017-12-20 12:35:18 -05001335 qdisc_put_rtab(qdisc_get_rtab(&hopt->ceil, tb[TCA_HTB_CTAB],
1336 NULL));
Jesper Dangaard Brouer8a8e3d82013-08-14 23:47:11 +02001337
Stephen Hemminger87990462006-08-10 23:35:16 -07001338 if (!cl) { /* new class */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001339 struct Qdisc *new_q;
Stephen Hemminger3696f622006-08-10 23:36:01 -07001340 int prio;
Patrick McHardyee39e102007-07-02 22:48:13 -07001341 struct {
Patrick McHardy1e904742008-01-22 22:11:17 -08001342 struct nlattr nla;
Patrick McHardyee39e102007-07-02 22:48:13 -07001343 struct gnet_estimator opt;
1344 } est = {
Patrick McHardy1e904742008-01-22 22:11:17 -08001345 .nla = {
1346 .nla_len = nla_attr_size(sizeof(est.opt)),
1347 .nla_type = TCA_RATE,
Patrick McHardyee39e102007-07-02 22:48:13 -07001348 },
1349 .opt = {
1350 /* 4s interval, 16s averaging constant */
1351 .interval = 2,
1352 .ewma_log = 2,
1353 },
1354 };
Stephen Hemminger3696f622006-08-10 23:36:01 -07001355
Linus Torvalds1da177e2005-04-16 15:20:36 -07001356 /* check for valid classid */
Joe Perchesf64f9e72009-11-29 16:55:45 -08001357 if (!classid || TC_H_MAJ(classid ^ sch->handle) ||
1358 htb_find(classid, sch))
Linus Torvalds1da177e2005-04-16 15:20:36 -07001359 goto failure;
1360
1361 /* check maximal depth */
1362 if (parent && parent->parent && parent->parent->level < 2) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001363 pr_err("htb: tree is too deep\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001364 goto failure;
1365 }
1366 err = -ENOBUFS;
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001367 cl = kzalloc(sizeof(*cl), GFP_KERNEL);
1368 if (!cl)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001369 goto failure;
Stephen Hemminger87990462006-08-10 23:35:16 -07001370
Alexander Aring8d1a77f2017-12-20 12:35:19 -05001371 err = tcf_block_get(&cl->block, &cl->filter_list, sch, extack);
Jiri Pirko6529eab2017-05-17 11:07:55 +02001372 if (err) {
1373 kfree(cl);
1374 goto failure;
1375 }
Eric Dumazet64153ce2013-06-06 14:53:16 -07001376 if (htb_rate_est || tca[TCA_RATE]) {
John Fastabend22e0f8b2014-09-28 11:52:56 -07001377 err = gen_new_estimator(&cl->bstats, NULL,
1378 &cl->rate_est,
Eric Dumazetedb09eb2016-06-06 09:37:16 -07001379 NULL,
1380 qdisc_root_sleeping_running(sch),
Eric Dumazet64153ce2013-06-06 14:53:16 -07001381 tca[TCA_RATE] ? : &est.nla);
1382 if (err) {
Jiri Pirko6529eab2017-05-17 11:07:55 +02001383 tcf_block_put(cl->block);
Eric Dumazet64153ce2013-06-06 14:53:16 -07001384 kfree(cl);
1385 goto failure;
1386 }
Stephen Hemminger71bcb092008-11-25 21:13:31 -08001387 }
1388
Patrick McHardy42077592008-07-05 23:22:53 -07001389 cl->children = 0;
Stephen Hemminger3696f622006-08-10 23:36:01 -07001390 RB_CLEAR_NODE(&cl->pq_node);
1391
1392 for (prio = 0; prio < TC_HTB_NUMPRIO; prio++)
1393 RB_CLEAR_NODE(&cl->node[prio]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001394
1395 /* create leaf qdisc early because it uses kmalloc(GFP_KERNEL)
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001396 * so that can't be used inside of sch_tree_lock
1397 * -- thanks to Karlis Peisenieks
1398 */
Alexander Aringa38a9882017-12-20 12:35:21 -05001399 new_q = qdisc_create_dflt(sch->dev_queue, &pfifo_qdisc_ops,
1400 classid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001401 sch_tree_lock(sch);
1402 if (parent && !parent->level) {
1403 /* turn parent into inner node */
Paolo Abenie5f0e8f2019-03-28 16:53:13 +01001404 qdisc_purge_queue(parent->leaf.q);
Vlad Buslov4ce70b42019-09-24 18:51:16 +03001405 parent_qdisc = parent->leaf.q;
Stephen Hemminger87990462006-08-10 23:35:16 -07001406 if (parent->prio_activity)
1407 htb_deactivate(q, parent);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001408
1409 /* remove from evt list because of level change */
1410 if (parent->cmode != HTB_CAN_SEND) {
Eric Dumazetc9364632013-06-15 03:30:10 -07001411 htb_safe_rb_erase(&parent->pq_node, &q->hlevel[0].wait_pq);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001412 parent->cmode = HTB_CAN_SEND;
1413 }
1414 parent->level = (parent->parent ? parent->parent->level
Stephen Hemminger87990462006-08-10 23:35:16 -07001415 : TC_HTB_MAXDEPTH) - 1;
Cong Wang11957be2018-09-07 13:29:14 -07001416 memset(&parent->inner, 0, sizeof(parent->inner));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001417 }
1418 /* leaf (we) needs elementary qdisc */
Cong Wang11957be2018-09-07 13:29:14 -07001419 cl->leaf.q = new_q ? new_q : &noop_qdisc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001420
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001421 cl->common.classid = classid;
Stephen Hemminger87990462006-08-10 23:35:16 -07001422 cl->parent = parent;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001423
1424 /* set class to be in HTB_CAN_SEND state */
Jiri Pirkob9a7afd2013-02-12 00:12:02 +00001425 cl->tokens = PSCHED_TICKS2NS(hopt->buffer);
1426 cl->ctokens = PSCHED_TICKS2NS(hopt->cbuffer);
Eric Dumazet5343a7f2013-06-04 07:11:48 +00001427 cl->mbuffer = 60ULL * NSEC_PER_SEC; /* 1min */
Eric Dumazetd2de8752014-08-22 18:32:09 -07001428 cl->t_c = ktime_get_ns();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001429 cl->cmode = HTB_CAN_SEND;
1430
1431 /* attach to the hash list and parent's family */
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001432 qdisc_class_hash_insert(&q->clhash, &cl->common);
Patrick McHardy42077592008-07-05 23:22:53 -07001433 if (parent)
1434 parent->children++;
Cong Wang11957be2018-09-07 13:29:14 -07001435 if (cl->leaf.q != &noop_qdisc)
1436 qdisc_hash_add(cl->leaf.q, true);
Patrick McHardyee39e102007-07-02 22:48:13 -07001437 } else {
Stephen Hemminger71bcb092008-11-25 21:13:31 -08001438 if (tca[TCA_RATE]) {
John Fastabend22e0f8b2014-09-28 11:52:56 -07001439 err = gen_replace_estimator(&cl->bstats, NULL,
1440 &cl->rate_est,
Eric Dumazetedb09eb2016-06-06 09:37:16 -07001441 NULL,
1442 qdisc_root_sleeping_running(sch),
Stephen Hemminger71bcb092008-11-25 21:13:31 -08001443 tca[TCA_RATE]);
1444 if (err)
1445 return err;
1446 }
Stephen Hemminger87990462006-08-10 23:35:16 -07001447 sch_tree_lock(sch);
Patrick McHardyee39e102007-07-02 22:48:13 -07001448 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001449
Yang Yingliang1598f7c2013-12-10 14:59:28 +08001450 rate64 = tb[TCA_HTB_RATE64] ? nla_get_u64(tb[TCA_HTB_RATE64]) : 0;
1451
1452 ceil64 = tb[TCA_HTB_CEIL64] ? nla_get_u64(tb[TCA_HTB_CEIL64]) : 0;
1453
1454 psched_ratecfg_precompute(&cl->rate, &hopt->rate, rate64);
1455 psched_ratecfg_precompute(&cl->ceil, &hopt->ceil, ceil64);
1456
Linus Torvalds1da177e2005-04-16 15:20:36 -07001457 /* it used to be a nasty bug here, we have to check that node
Cong Wang11957be2018-09-07 13:29:14 -07001458 * is really leaf before changing cl->leaf !
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001459 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001460 if (!cl->level) {
Yang Yingliang1598f7c2013-12-10 14:59:28 +08001461 u64 quantum = cl->rate.rate_bytes_ps;
1462
1463 do_div(quantum, q->rate2quantum);
1464 cl->quantum = min_t(u64, quantum, INT_MAX);
1465
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001466 if (!hopt->quantum && cl->quantum < 1000) {
Li RongQingda01ec42018-03-30 10:11:21 +08001467 warn = -1;
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001468 cl->quantum = 1000;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001469 }
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001470 if (!hopt->quantum && cl->quantum > 200000) {
Li RongQingda01ec42018-03-30 10:11:21 +08001471 warn = 1;
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001472 cl->quantum = 200000;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001473 }
1474 if (hopt->quantum)
Jarek Poplawskic19f7a32008-12-03 21:09:45 -08001475 cl->quantum = hopt->quantum;
1476 if ((cl->prio = hopt->prio) >= TC_HTB_NUMPRIO)
1477 cl->prio = TC_HTB_NUMPRIO - 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001478 }
1479
Jiri Pirko324f5aa2013-02-12 00:11:59 +00001480 cl->buffer = PSCHED_TICKS2NS(hopt->buffer);
Vimalkumarf3ad8572013-09-10 17:36:37 -07001481 cl->cbuffer = PSCHED_TICKS2NS(hopt->cbuffer);
Vimalkumar56b765b2012-10-31 06:04:11 +00001482
Linus Torvalds1da177e2005-04-16 15:20:36 -07001483 sch_tree_unlock(sch);
Vlad Buslov4ce70b42019-09-24 18:51:16 +03001484 qdisc_put(parent_qdisc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001485
Li RongQingda01ec42018-03-30 10:11:21 +08001486 if (warn)
1487 pr_warn("HTB: quantum of class %X is %s. Consider r2q change.\n",
1488 cl->common.classid, (warn == -1 ? "small" : "big"));
1489
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001490 qdisc_class_hash_grow(sch, &q->clhash);
1491
Linus Torvalds1da177e2005-04-16 15:20:36 -07001492 *arg = (unsigned long)cl;
1493 return 0;
1494
1495failure:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001496 return err;
1497}
1498
Alexander Aringcbaacc42017-12-20 12:35:16 -05001499static struct tcf_block *htb_tcf_block(struct Qdisc *sch, unsigned long arg,
1500 struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001501{
1502 struct htb_sched *q = qdisc_priv(sch);
1503 struct htb_class *cl = (struct htb_class *)arg;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001504
Jiri Pirko6529eab2017-05-17 11:07:55 +02001505 return cl ? cl->block : q->block;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001506}
1507
1508static unsigned long htb_bind_filter(struct Qdisc *sch, unsigned long parent,
Stephen Hemminger87990462006-08-10 23:35:16 -07001509 u32 classid)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001510{
Stephen Hemminger87990462006-08-10 23:35:16 -07001511 struct htb_class *cl = htb_find(classid, sch);
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001512
Linus Torvalds1da177e2005-04-16 15:20:36 -07001513 /*if (cl && !cl->level) return 0;
Eric Dumazetcc7ec452011-01-19 19:26:56 +00001514 * The line above used to be there to prevent attaching filters to
1515 * leaves. But at least tc_index filter uses this just to get class
1516 * for other reasons so that we have to allow for it.
1517 * ----
1518 * 19.6.2002 As Werner explained it is ok - bind filter is just
1519 * another way to "lock" the class - unlike "get" this lock can
1520 * be broken by class during destroy IIUC.
Linus Torvalds1da177e2005-04-16 15:20:36 -07001521 */
Stephen Hemminger87990462006-08-10 23:35:16 -07001522 if (cl)
1523 cl->filter_cnt++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001524 return (unsigned long)cl;
1525}
1526
1527static void htb_unbind_filter(struct Qdisc *sch, unsigned long arg)
1528{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001529 struct htb_class *cl = (struct htb_class *)arg;
Stephen Hemminger3bf72952006-08-10 23:31:08 -07001530
Stephen Hemminger87990462006-08-10 23:35:16 -07001531 if (cl)
1532 cl->filter_cnt--;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001533}
1534
1535static void htb_walk(struct Qdisc *sch, struct qdisc_walker *arg)
1536{
1537 struct htb_sched *q = qdisc_priv(sch);
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001538 struct htb_class *cl;
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001539 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001540
1541 if (arg->stop)
1542 return;
1543
Patrick McHardyf4c1f3e2008-07-05 23:22:35 -07001544 for (i = 0; i < q->clhash.hashsize; i++) {
Sasha Levinb67bfe02013-02-27 17:06:00 -08001545 hlist_for_each_entry(cl, &q->clhash.hash[i], common.hnode) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001546 if (arg->count < arg->skip) {
1547 arg->count++;
1548 continue;
1549 }
1550 if (arg->fn(sch, (unsigned long)cl, arg) < 0) {
1551 arg->stop = 1;
1552 return;
1553 }
1554 arg->count++;
1555 }
1556 }
1557}
1558
Eric Dumazet20fea082007-11-14 01:44:41 -08001559static const struct Qdisc_class_ops htb_class_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001560 .graft = htb_graft,
1561 .leaf = htb_leaf,
Patrick McHardy256d61b2006-11-29 17:37:05 -08001562 .qlen_notify = htb_qlen_notify,
WANG Cong143976c2017-08-24 16:51:29 -07001563 .find = htb_search,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001564 .change = htb_change_class,
1565 .delete = htb_delete,
1566 .walk = htb_walk,
Jiri Pirko6529eab2017-05-17 11:07:55 +02001567 .tcf_block = htb_tcf_block,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001568 .bind_tcf = htb_bind_filter,
1569 .unbind_tcf = htb_unbind_filter,
1570 .dump = htb_dump_class,
1571 .dump_stats = htb_dump_class_stats,
1572};
1573
Eric Dumazet20fea082007-11-14 01:44:41 -08001574static struct Qdisc_ops htb_qdisc_ops __read_mostly = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001575 .cl_ops = &htb_class_ops,
1576 .id = "htb",
1577 .priv_size = sizeof(struct htb_sched),
1578 .enqueue = htb_enqueue,
1579 .dequeue = htb_dequeue,
Jarek Poplawski77be1552008-10-31 00:47:01 -07001580 .peek = qdisc_peek_dequeued,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001581 .init = htb_init,
1582 .reset = htb_reset,
1583 .destroy = htb_destroy,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001584 .dump = htb_dump,
1585 .owner = THIS_MODULE,
1586};
1587
1588static int __init htb_module_init(void)
1589{
Stephen Hemminger87990462006-08-10 23:35:16 -07001590 return register_qdisc(&htb_qdisc_ops);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001591}
Stephen Hemminger87990462006-08-10 23:35:16 -07001592static void __exit htb_module_exit(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001593{
Stephen Hemminger87990462006-08-10 23:35:16 -07001594 unregister_qdisc(&htb_qdisc_ops);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001595}
Stephen Hemminger87990462006-08-10 23:35:16 -07001596
Linus Torvalds1da177e2005-04-16 15:20:36 -07001597module_init(htb_module_init)
1598module_exit(htb_module_exit)
1599MODULE_LICENSE("GPL");