blob: 555a1b9e467fc5f5cbd28944510c7dfd4527637d [file] [log] [blame]
Thomas Gleixner2874c5f2019-05-27 08:55:01 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Linus Torvalds1da177e2005-04-16 15:20:36 -07002/*
3 * net/sched/sch_red.c Random Early Detection queue.
4 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 * Authors: Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru>
6 *
7 * Changes:
Thomas Grafdba051f2005-11-05 21:14:08 +01008 * J Hadi Salim 980914: computation fixes
Linus Torvalds1da177e2005-04-16 15:20:36 -07009 * Alexey Makarenko <makar@phoenix.kharkov.ua> 990814: qave on idle link was calculated incorrectly.
Thomas Grafdba051f2005-11-05 21:14:08 +010010 * J Hadi Salim 980816: ECN support
Linus Torvalds1da177e2005-04-16 15:20:36 -070011 */
12
Linus Torvalds1da177e2005-04-16 15:20:36 -070013#include <linux/module.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070014#include <linux/types.h>
15#include <linux/kernel.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070016#include <linux/skbuff.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070017#include <net/pkt_sched.h>
Nogah Frankel602f3ba2017-11-06 07:23:41 +010018#include <net/pkt_cls.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070019#include <net/inet_ecn.h>
Thomas Graf6b31b282005-11-05 21:14:05 +010020#include <net/red.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070021
22
Thomas Graf6b31b282005-11-05 21:14:05 +010023/* Parameters, settable by user:
Linus Torvalds1da177e2005-04-16 15:20:36 -070024 -----------------------------
25
26 limit - bytes (must be > qth_max + burst)
27
28 Hard limit on queue length, should be chosen >qth_max
29 to allow packet bursts. This parameter does not
30 affect the algorithms behaviour and can be chosen
31 arbitrarily high (well, less than ram size)
32 Really, this limit will never be reached
33 if RED works correctly.
Linus Torvalds1da177e2005-04-16 15:20:36 -070034 */
35
Eric Dumazetcc7ec452011-01-19 19:26:56 +000036struct red_sched_data {
Thomas Graf6b31b282005-11-05 21:14:05 +010037 u32 limit; /* HARD maximal queue length */
Petr Machata14bc1752020-03-13 01:10:56 +020038
Thomas Graf6b31b282005-11-05 21:14:05 +010039 unsigned char flags;
Petr Machata14bc1752020-03-13 01:10:56 +020040 /* Non-flags in tc_red_qopt.flags. */
41 unsigned char userbits;
42
Eric Dumazet8af2a212011-12-08 06:06:03 +000043 struct timer_list adapt_timer;
Kees Cookcdeabbb2017-10-16 17:29:17 -070044 struct Qdisc *sch;
Thomas Graf6b31b282005-11-05 21:14:05 +010045 struct red_parms parms;
Eric Dumazeteeca6682012-01-05 02:25:16 +000046 struct red_vars vars;
Thomas Graf6b31b282005-11-05 21:14:05 +010047 struct red_stats stats;
Patrick McHardyf38c39d2006-03-20 19:20:44 -080048 struct Qdisc *qdisc;
Linus Torvalds1da177e2005-04-16 15:20:36 -070049};
50
Johannes Berg47a14942020-04-30 22:13:05 +020051#define TC_RED_SUPPORTED_FLAGS (TC_RED_HISTORIC_FLAGS | TC_RED_NODROP)
Petr Machata14bc1752020-03-13 01:10:56 +020052
Thomas Graf6b31b282005-11-05 21:14:05 +010053static inline int red_use_ecn(struct red_sched_data *q)
Linus Torvalds1da177e2005-04-16 15:20:36 -070054{
Thomas Graf6b31b282005-11-05 21:14:05 +010055 return q->flags & TC_RED_ECN;
Linus Torvalds1da177e2005-04-16 15:20:36 -070056}
57
Thomas Grafbdc450a2005-11-05 21:14:28 +010058static inline int red_use_harddrop(struct red_sched_data *q)
59{
60 return q->flags & TC_RED_HARDDROP;
61}
62
Petr Machata0a7fad22020-03-13 01:10:57 +020063static int red_use_nodrop(struct red_sched_data *q)
64{
65 return q->flags & TC_RED_NODROP;
66}
67
Eric Dumazet520ac302016-06-21 23:16:49 -070068static int red_enqueue(struct sk_buff *skb, struct Qdisc *sch,
69 struct sk_buff **to_free)
Linus Torvalds1da177e2005-04-16 15:20:36 -070070{
71 struct red_sched_data *q = qdisc_priv(sch);
Patrick McHardyf38c39d2006-03-20 19:20:44 -080072 struct Qdisc *child = q->qdisc;
73 int ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -070074
Eric Dumazeteeca6682012-01-05 02:25:16 +000075 q->vars.qavg = red_calc_qavg(&q->parms,
76 &q->vars,
77 child->qstats.backlog);
Linus Torvalds1da177e2005-04-16 15:20:36 -070078
Eric Dumazeteeca6682012-01-05 02:25:16 +000079 if (red_is_idling(&q->vars))
80 red_end_of_idle_period(&q->vars);
Linus Torvalds1da177e2005-04-16 15:20:36 -070081
Eric Dumazeteeca6682012-01-05 02:25:16 +000082 switch (red_action(&q->parms, &q->vars, q->vars.qavg)) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +000083 case RED_DONT_MARK:
84 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -070085
Eric Dumazetcc7ec452011-01-19 19:26:56 +000086 case RED_PROB_MARK:
John Fastabend25331d62014-09-28 11:53:29 -070087 qdisc_qstats_overlimit(sch);
Petr Machata0a7fad22020-03-13 01:10:57 +020088 if (!red_use_ecn(q)) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +000089 q->stats.prob_drop++;
90 goto congestion_drop;
91 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070092
Petr Machata0a7fad22020-03-13 01:10:57 +020093 if (INET_ECN_set_ce(skb)) {
94 q->stats.prob_mark++;
95 } else if (!red_use_nodrop(q)) {
96 q->stats.prob_drop++;
97 goto congestion_drop;
98 }
99
100 /* Non-ECT packet in ECN nodrop mode: queue it. */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000101 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700102
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000103 case RED_HARD_MARK:
John Fastabend25331d62014-09-28 11:53:29 -0700104 qdisc_qstats_overlimit(sch);
Petr Machata0a7fad22020-03-13 01:10:57 +0200105 if (red_use_harddrop(q) || !red_use_ecn(q)) {
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000106 q->stats.forced_drop++;
107 goto congestion_drop;
108 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700109
Petr Machata0a7fad22020-03-13 01:10:57 +0200110 if (INET_ECN_set_ce(skb)) {
111 q->stats.forced_mark++;
112 } else if (!red_use_nodrop(q)) {
113 q->stats.forced_drop++;
114 goto congestion_drop;
115 }
116
117 /* Non-ECT packet in ECN nodrop mode: queue it. */
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000118 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700119 }
120
Eric Dumazet520ac302016-06-21 23:16:49 -0700121 ret = qdisc_enqueue(skb, child, to_free);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800122 if (likely(ret == NET_XMIT_SUCCESS)) {
WANG Congd7f4f332016-06-01 16:15:18 -0700123 qdisc_qstats_backlog_inc(sch, skb);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800124 sch->q.qlen++;
Jarek Poplawski378a2f02008-08-04 22:31:03 -0700125 } else if (net_xmit_drop_count(ret)) {
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800126 q->stats.pdrop++;
John Fastabend25331d62014-09-28 11:53:29 -0700127 qdisc_qstats_drop(sch);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800128 }
129 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700130
Thomas Graf6b31b282005-11-05 21:14:05 +0100131congestion_drop:
Eric Dumazet520ac302016-06-21 23:16:49 -0700132 qdisc_drop(skb, sch, to_free);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700133 return NET_XMIT_CN;
134}
135
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000136static struct sk_buff *red_dequeue(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700137{
138 struct sk_buff *skb;
139 struct red_sched_data *q = qdisc_priv(sch);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800140 struct Qdisc *child = q->qdisc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700141
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800142 skb = child->dequeue(child);
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800143 if (skb) {
144 qdisc_bstats_update(sch, skb);
WANG Congd7f4f332016-06-01 16:15:18 -0700145 qdisc_qstats_backlog_dec(sch, skb);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800146 sch->q.qlen--;
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800147 } else {
Eric Dumazeteeca6682012-01-05 02:25:16 +0000148 if (!red_is_idling(&q->vars))
149 red_start_of_idle_period(&q->vars);
Eric Dumazet9190b3b2011-01-20 23:31:33 -0800150 }
Thomas Graf9e178ff2005-11-05 21:14:06 +0100151 return skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700152}
153
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000154static struct sk_buff *red_peek(struct Qdisc *sch)
Jarek Poplawski8e3af972008-10-31 00:45:55 -0700155{
156 struct red_sched_data *q = qdisc_priv(sch);
157 struct Qdisc *child = q->qdisc;
158
159 return child->ops->peek(child);
160}
161
Eric Dumazetcc7ec452011-01-19 19:26:56 +0000162static void red_reset(struct Qdisc *sch)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700163{
164 struct red_sched_data *q = qdisc_priv(sch);
165
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800166 qdisc_reset(q->qdisc);
WANG Congd7f4f332016-06-01 16:15:18 -0700167 sch->qstats.backlog = 0;
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800168 sch->q.qlen = 0;
Eric Dumazeteeca6682012-01-05 02:25:16 +0000169 red_restart(&q->vars);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700170}
171
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100172static int red_offload(struct Qdisc *sch, bool enable)
173{
174 struct red_sched_data *q = qdisc_priv(sch);
175 struct net_device *dev = qdisc_dev(sch);
176 struct tc_red_qopt_offload opt = {
177 .handle = sch->handle,
178 .parent = sch->parent,
179 };
180
181 if (!tc_can_offload(dev) || !dev->netdev_ops->ndo_setup_tc)
182 return -EOPNOTSUPP;
183
184 if (enable) {
185 opt.command = TC_RED_REPLACE;
186 opt.set.min = q->parms.qth_min >> q->parms.Wlog;
187 opt.set.max = q->parms.qth_max >> q->parms.Wlog;
188 opt.set.probability = q->parms.max_P;
Jakub Kicinskic0b74902018-11-12 14:58:16 -0800189 opt.set.limit = q->limit;
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100190 opt.set.is_ecn = red_use_ecn(q);
Jakub Kicinski190852a2018-11-08 19:50:38 -0800191 opt.set.is_harddrop = red_use_harddrop(q);
Petr Machata0a7fad22020-03-13 01:10:57 +0200192 opt.set.is_nodrop = red_use_nodrop(q);
Jakub Kicinski416ef9b2018-01-14 20:01:26 -0800193 opt.set.qstats = &sch->qstats;
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100194 } else {
195 opt.command = TC_RED_DESTROY;
196 }
197
Nogah Frankel8234af22017-12-25 10:51:41 +0200198 return dev->netdev_ops->ndo_setup_tc(dev, TC_SETUP_QDISC_RED, &opt);
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100199}
200
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800201static void red_destroy(struct Qdisc *sch)
202{
203 struct red_sched_data *q = qdisc_priv(sch);
Eric Dumazet8af2a212011-12-08 06:06:03 +0000204
205 del_timer_sync(&q->adapt_timer);
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100206 red_offload(sch, false);
Vlad Buslov86bd4462018-09-24 19:22:50 +0300207 qdisc_put(q->qdisc);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800208}
209
Patrick McHardy27a34212008-01-23 20:35:39 -0800210static const struct nla_policy red_policy[TCA_RED_MAX + 1] = {
Petr Machata14bc1752020-03-13 01:10:56 +0200211 [TCA_RED_UNSPEC] = { .strict_start_type = TCA_RED_FLAGS },
Patrick McHardy27a34212008-01-23 20:35:39 -0800212 [TCA_RED_PARMS] = { .len = sizeof(struct tc_red_qopt) },
213 [TCA_RED_STAB] = { .len = RED_STAB_SIZE },
Eric Dumazeta73ed262011-12-09 02:46:45 +0000214 [TCA_RED_MAX_P] = { .type = NLA_U32 },
Johannes Berg47a14942020-04-30 22:13:05 +0200215 [TCA_RED_FLAGS] = NLA_POLICY_BITFIELD32(TC_RED_SUPPORTED_FLAGS),
Patrick McHardy27a34212008-01-23 20:35:39 -0800216};
217
Alexander Aring20307212017-12-20 12:35:14 -0500218static int red_change(struct Qdisc *sch, struct nlattr *opt,
219 struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700220{
Jakub Kicinski0c8d13a2018-11-07 17:33:39 -0800221 struct Qdisc *old_child = NULL, *child = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700222 struct red_sched_data *q = qdisc_priv(sch);
Patrick McHardy1e904742008-01-22 22:11:17 -0800223 struct nlattr *tb[TCA_RED_MAX + 1];
Petr Machata14bc1752020-03-13 01:10:56 +0200224 struct nla_bitfield32 flags_bf;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700225 struct tc_red_qopt *ctl;
Petr Machata14bc1752020-03-13 01:10:56 +0200226 unsigned char userbits;
227 unsigned char flags;
Patrick McHardycee63722008-01-23 20:33:32 -0800228 int err;
Eric Dumazeta73ed262011-12-09 02:46:45 +0000229 u32 max_P;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700230
Patrick McHardycee63722008-01-23 20:33:32 -0800231 if (opt == NULL)
Thomas Grafdba051f2005-11-05 21:14:08 +0100232 return -EINVAL;
233
Johannes Berg8cb08172019-04-26 14:07:28 +0200234 err = nla_parse_nested_deprecated(tb, TCA_RED_MAX, opt, red_policy,
235 NULL);
Patrick McHardycee63722008-01-23 20:33:32 -0800236 if (err < 0)
237 return err;
238
Patrick McHardy1e904742008-01-22 22:11:17 -0800239 if (tb[TCA_RED_PARMS] == NULL ||
Patrick McHardy27a34212008-01-23 20:35:39 -0800240 tb[TCA_RED_STAB] == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700241 return -EINVAL;
242
Eric Dumazeta73ed262011-12-09 02:46:45 +0000243 max_P = tb[TCA_RED_MAX_P] ? nla_get_u32(tb[TCA_RED_MAX_P]) : 0;
244
Patrick McHardy1e904742008-01-22 22:11:17 -0800245 ctl = nla_data(tb[TCA_RED_PARMS]);
Nogah Frankel8afa10c2017-12-04 13:31:11 +0200246 if (!red_check_params(ctl->qth_min, ctl->qth_max, ctl->Wlog))
247 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700248
Petr Machata14bc1752020-03-13 01:10:56 +0200249 err = red_get_flags(ctl->flags, TC_RED_HISTORIC_FLAGS,
Johannes Berg47a14942020-04-30 22:13:05 +0200250 tb[TCA_RED_FLAGS], TC_RED_SUPPORTED_FLAGS,
Petr Machata14bc1752020-03-13 01:10:56 +0200251 &flags_bf, &userbits, extack);
252 if (err)
253 return err;
254
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800255 if (ctl->limit > 0) {
Alexander Aringa38a98822017-12-20 12:35:21 -0500256 child = fifo_create_dflt(sch, &bfifo_qdisc_ops, ctl->limit,
257 extack);
Patrick McHardyfb0305c2008-07-05 23:40:21 -0700258 if (IS_ERR(child))
259 return PTR_ERR(child);
Paolo Abeni44a63b132018-05-18 14:51:44 +0200260
261 /* child is fifo, no need to check for noop_qdisc */
262 qdisc_hash_add(child, true);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800263 }
264
Linus Torvalds1da177e2005-04-16 15:20:36 -0700265 sch_tree_lock(sch);
Petr Machata14bc1752020-03-13 01:10:56 +0200266
267 flags = (q->flags & ~flags_bf.selector) | flags_bf.value;
268 err = red_validate_flags(flags, extack);
269 if (err)
270 goto unlock_out;
271
272 q->flags = flags;
273 q->userbits = userbits;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700274 q->limit = ctl->limit;
Patrick McHardy5e50da02006-11-29 17:36:20 -0800275 if (child) {
Paolo Abenie5f0e8f2019-03-28 16:53:13 +0100276 qdisc_tree_flush_backlog(q->qdisc);
Jakub Kicinski0c8d13a2018-11-07 17:33:39 -0800277 old_child = q->qdisc;
Patrick McHardyb94c8af2008-11-20 04:11:36 -0800278 q->qdisc = child;
Patrick McHardy5e50da02006-11-29 17:36:20 -0800279 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700280
Eric Dumazeteeca6682012-01-05 02:25:16 +0000281 red_set_parms(&q->parms,
282 ctl->qth_min, ctl->qth_max, ctl->Wlog,
Eric Dumazeta73ed262011-12-09 02:46:45 +0000283 ctl->Plog, ctl->Scell_log,
284 nla_data(tb[TCA_RED_STAB]),
285 max_P);
Eric Dumazeteeca6682012-01-05 02:25:16 +0000286 red_set_vars(&q->vars);
Thomas Graf6b31b282005-11-05 21:14:05 +0100287
Eric Dumazet8af2a212011-12-08 06:06:03 +0000288 del_timer(&q->adapt_timer);
289 if (ctl->flags & TC_RED_ADAPTATIVE)
290 mod_timer(&q->adapt_timer, jiffies + HZ/2);
291
Eric Dumazet1ee5fa12011-12-01 11:06:34 +0000292 if (!q->qdisc->q.qlen)
Eric Dumazeteeca6682012-01-05 02:25:16 +0000293 red_start_of_idle_period(&q->vars);
Thomas Grafdba051f2005-11-05 21:14:08 +0100294
Linus Torvalds1da177e2005-04-16 15:20:36 -0700295 sch_tree_unlock(sch);
Jakub Kicinski0c8d13a2018-11-07 17:33:39 -0800296
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100297 red_offload(sch, true);
Jakub Kicinski0c8d13a2018-11-07 17:33:39 -0800298
299 if (old_child)
300 qdisc_put(old_child);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700301 return 0;
Petr Machata14bc1752020-03-13 01:10:56 +0200302
303unlock_out:
304 sch_tree_unlock(sch);
305 if (child)
306 qdisc_put(child);
307 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700308}
309
Kees Cookcdeabbb2017-10-16 17:29:17 -0700310static inline void red_adaptative_timer(struct timer_list *t)
Eric Dumazet8af2a212011-12-08 06:06:03 +0000311{
Kees Cookcdeabbb2017-10-16 17:29:17 -0700312 struct red_sched_data *q = from_timer(q, t, adapt_timer);
313 struct Qdisc *sch = q->sch;
Eric Dumazet8af2a212011-12-08 06:06:03 +0000314 spinlock_t *root_lock = qdisc_lock(qdisc_root_sleeping(sch));
315
316 spin_lock(root_lock);
Eric Dumazeteeca6682012-01-05 02:25:16 +0000317 red_adaptative_algo(&q->parms, &q->vars);
Eric Dumazet8af2a212011-12-08 06:06:03 +0000318 mod_timer(&q->adapt_timer, jiffies + HZ/2);
319 spin_unlock(root_lock);
320}
321
Alexander Aringe63d7dfd2017-12-20 12:35:13 -0500322static int red_init(struct Qdisc *sch, struct nlattr *opt,
323 struct netlink_ext_ack *extack)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700324{
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800325 struct red_sched_data *q = qdisc_priv(sch);
326
327 q->qdisc = &noop_qdisc;
Kees Cookcdeabbb2017-10-16 17:29:17 -0700328 q->sch = sch;
329 timer_setup(&q->adapt_timer, red_adaptative_timer, 0);
Alexander Aring20307212017-12-20 12:35:14 -0500330 return red_change(sch, opt, extack);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700331}
332
Jakub Kicinskidad54c02018-11-07 17:33:35 -0800333static int red_dump_offload_stats(struct Qdisc *sch)
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100334{
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100335 struct tc_red_qopt_offload hw_stats = {
Andrew Mortonee9d3422017-11-10 15:09:53 -0800336 .command = TC_RED_STATS,
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100337 .handle = sch->handle,
338 .parent = sch->parent,
Andrew Mortonee9d3422017-11-10 15:09:53 -0800339 {
340 .stats.bstats = &sch->bstats,
341 .stats.qstats = &sch->qstats,
342 },
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100343 };
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100344
Jakub Kicinskib5928432018-11-07 17:33:34 -0800345 return qdisc_offload_dump_helper(sch, TC_SETUP_QDISC_RED, &hw_stats);
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100346}
347
Linus Torvalds1da177e2005-04-16 15:20:36 -0700348static int red_dump(struct Qdisc *sch, struct sk_buff *skb)
349{
350 struct red_sched_data *q = qdisc_priv(sch);
Patrick McHardy1e904742008-01-22 22:11:17 -0800351 struct nlattr *opts = NULL;
Thomas Graf6b31b282005-11-05 21:14:05 +0100352 struct tc_red_qopt opt = {
353 .limit = q->limit,
Petr Machata14bc1752020-03-13 01:10:56 +0200354 .flags = (q->flags & TC_RED_HISTORIC_FLAGS) |
355 q->userbits,
Thomas Graf6b31b282005-11-05 21:14:05 +0100356 .qth_min = q->parms.qth_min >> q->parms.Wlog,
357 .qth_max = q->parms.qth_max >> q->parms.Wlog,
358 .Wlog = q->parms.Wlog,
359 .Plog = q->parms.Plog,
360 .Scell_log = q->parms.Scell_log,
361 };
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100362 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700363
Jakub Kicinskidad54c02018-11-07 17:33:35 -0800364 err = red_dump_offload_stats(sch);
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100365 if (err)
366 goto nla_put_failure;
367
Michal Kubecekae0be8d2019-04-26 11:13:06 +0200368 opts = nla_nest_start_noflag(skb, TCA_OPTIONS);
Patrick McHardy1e904742008-01-22 22:11:17 -0800369 if (opts == NULL)
370 goto nla_put_failure;
David S. Miller1b34ec42012-03-29 05:11:39 -0400371 if (nla_put(skb, TCA_RED_PARMS, sizeof(opt), &opt) ||
Petr Machata14bc1752020-03-13 01:10:56 +0200372 nla_put_u32(skb, TCA_RED_MAX_P, q->parms.max_P) ||
Jiri Pirko8953b072020-03-28 16:37:42 +0100373 nla_put_bitfield32(skb, TCA_RED_FLAGS,
Johannes Berg47a14942020-04-30 22:13:05 +0200374 q->flags, TC_RED_SUPPORTED_FLAGS))
David S. Miller1b34ec42012-03-29 05:11:39 -0400375 goto nla_put_failure;
Patrick McHardy1e904742008-01-22 22:11:17 -0800376 return nla_nest_end(skb, opts);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700377
Patrick McHardy1e904742008-01-22 22:11:17 -0800378nla_put_failure:
Thomas Grafbc3ed282008-06-03 16:36:54 -0700379 nla_nest_cancel(skb, opts);
380 return -EMSGSIZE;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700381}
382
383static int red_dump_stats(struct Qdisc *sch, struct gnet_dump *d)
384{
385 struct red_sched_data *q = qdisc_priv(sch);
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100386 struct net_device *dev = qdisc_dev(sch);
Nogah Frankelf8253df2018-01-10 14:59:59 +0100387 struct tc_red_xstats st = {0};
Linus Torvalds1da177e2005-04-16 15:20:36 -0700388
Yuval Mintz428a68a2017-12-14 15:54:30 +0200389 if (sch->flags & TCQ_F_OFFLOADED) {
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100390 struct tc_red_qopt_offload hw_stats_request = {
Andrew Mortonee9d3422017-11-10 15:09:53 -0800391 .command = TC_RED_XSTATS,
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100392 .handle = sch->handle,
393 .parent = sch->parent,
Andrew Mortonee9d3422017-11-10 15:09:53 -0800394 {
Nogah Frankelf8253df2018-01-10 14:59:59 +0100395 .xstats = &q->stats,
Andrew Mortonee9d3422017-11-10 15:09:53 -0800396 },
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100397 };
Nogah Frankelf8253df2018-01-10 14:59:59 +0100398 dev->netdev_ops->ndo_setup_tc(dev, TC_SETUP_QDISC_RED,
399 &hw_stats_request);
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100400 }
Nogah Frankelf8253df2018-01-10 14:59:59 +0100401 st.early = q->stats.prob_drop + q->stats.forced_drop;
402 st.pdrop = q->stats.pdrop;
403 st.other = q->stats.other;
404 st.marked = q->stats.prob_mark + q->stats.forced_mark;
Nogah Frankel602f3ba2017-11-06 07:23:41 +0100405
Thomas Graf6b31b282005-11-05 21:14:05 +0100406 return gnet_stats_copy_app(d, &st, sizeof(st));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700407}
408
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800409static int red_dump_class(struct Qdisc *sch, unsigned long cl,
410 struct sk_buff *skb, struct tcmsg *tcm)
411{
412 struct red_sched_data *q = qdisc_priv(sch);
413
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800414 tcm->tcm_handle |= TC_H_MIN(1);
415 tcm->tcm_info = q->qdisc->handle;
416 return 0;
417}
418
Jakub Kicinskibf2a7522018-11-12 14:58:13 -0800419static void red_graft_offload(struct Qdisc *sch,
420 struct Qdisc *new, struct Qdisc *old,
421 struct netlink_ext_ack *extack)
422{
423 struct tc_red_qopt_offload graft_offload = {
424 .handle = sch->handle,
425 .parent = sch->parent,
426 .child_handle = new->handle,
427 .command = TC_RED_GRAFT,
428 };
429
430 qdisc_offload_graft_helper(qdisc_dev(sch), sch, new, old,
431 TC_SETUP_QDISC_RED, &graft_offload, extack);
432}
433
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800434static int red_graft(struct Qdisc *sch, unsigned long arg, struct Qdisc *new,
Alexander Aring653d6fd2017-12-20 12:35:17 -0500435 struct Qdisc **old, struct netlink_ext_ack *extack)
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800436{
437 struct red_sched_data *q = qdisc_priv(sch);
438
439 if (new == NULL)
440 new = &noop_qdisc;
441
WANG Cong86a79962016-02-25 14:55:00 -0800442 *old = qdisc_replace(sch, new, &q->qdisc);
Jakub Kicinskibf2a7522018-11-12 14:58:13 -0800443
444 red_graft_offload(sch, new, *old, extack);
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800445 return 0;
446}
447
448static struct Qdisc *red_leaf(struct Qdisc *sch, unsigned long arg)
449{
450 struct red_sched_data *q = qdisc_priv(sch);
451 return q->qdisc;
452}
453
WANG Cong143976c2017-08-24 16:51:29 -0700454static unsigned long red_find(struct Qdisc *sch, u32 classid)
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800455{
456 return 1;
457}
458
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800459static void red_walk(struct Qdisc *sch, struct qdisc_walker *walker)
460{
461 if (!walker->stop) {
462 if (walker->count >= walker->skip)
463 if (walker->fn(sch, 1, walker) < 0) {
464 walker->stop = 1;
465 return;
466 }
467 walker->count++;
468 }
469}
470
Eric Dumazet20fea082007-11-14 01:44:41 -0800471static const struct Qdisc_class_ops red_class_ops = {
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800472 .graft = red_graft,
473 .leaf = red_leaf,
WANG Cong143976c2017-08-24 16:51:29 -0700474 .find = red_find,
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800475 .walk = red_walk,
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800476 .dump = red_dump_class,
477};
478
Eric Dumazet20fea082007-11-14 01:44:41 -0800479static struct Qdisc_ops red_qdisc_ops __read_mostly = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700480 .id = "red",
481 .priv_size = sizeof(struct red_sched_data),
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800482 .cl_ops = &red_class_ops,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700483 .enqueue = red_enqueue,
484 .dequeue = red_dequeue,
Jarek Poplawski8e3af972008-10-31 00:45:55 -0700485 .peek = red_peek,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700486 .init = red_init,
487 .reset = red_reset,
Patrick McHardyf38c39d2006-03-20 19:20:44 -0800488 .destroy = red_destroy,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700489 .change = red_change,
490 .dump = red_dump,
491 .dump_stats = red_dump_stats,
492 .owner = THIS_MODULE,
493};
494
495static int __init red_module_init(void)
496{
497 return register_qdisc(&red_qdisc_ops);
498}
Thomas Grafdba051f2005-11-05 21:14:08 +0100499
500static void __exit red_module_exit(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700501{
502 unregister_qdisc(&red_qdisc_ops);
503}
Thomas Grafdba051f2005-11-05 21:14:08 +0100504
Linus Torvalds1da177e2005-04-16 15:20:36 -0700505module_init(red_module_init)
506module_exit(red_module_exit)
Thomas Grafdba051f2005-11-05 21:14:08 +0100507
Linus Torvalds1da177e2005-04-16 15:20:36 -0700508MODULE_LICENSE("GPL");