xref: /freebsd/sys/netpfil/ipfw/dn_aqm_codel.c (revision e6bfd18d21b225af6a0ed67ceeaf1293b7b9eba5)
1 /*
2  * Codel - The Controlled-Delay Active Queue Management algorithm.
3  *
4  * $FreeBSD$
5  *
6  * Copyright (C) 2016 Centre for Advanced Internet Architectures,
7  *  Swinburne University of Technology, Melbourne, Australia.
8  * Portions of this code were made possible in part by a gift from
9  *  The Comcast Innovation Fund.
10  * Implemented by Rasool Al-Saadi <ralsaadi@swin.edu.au>
11  *
12  * Redistribution and use in source and binary forms, with or without
13  * modification, are permitted provided that the following conditions
14  * are met:
15  * 1. Redistributions of source code must retain the above copyright
16  *    notice, this list of conditions and the following disclaimer.
17  * 2. Redistributions in binary form must reproduce the above copyright
18  *    notice, this list of conditions and the following disclaimer in the
19  *    documentation and/or other materials provided with the distribution.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
22  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24  * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
25  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31  * SUCH DAMAGE.
32  */
33 
34 #include <sys/cdefs.h>
35 #include "opt_inet6.h"
36 
37 #include <sys/param.h>
38 #include <sys/systm.h>
39 #include <sys/malloc.h>
40 #include <sys/mbuf.h>
41 #include <sys/kernel.h>
42 #include <sys/lock.h>
43 #include <sys/module.h>
44 #include <sys/priv.h>
45 #include <sys/proc.h>
46 #include <sys/rwlock.h>
47 #include <sys/socket.h>
48 #include <sys/time.h>
49 #include <sys/sysctl.h>
50 
51 #include <net/if.h>	/* IFNAMSIZ, struct ifaddr, ifq head, lock.h mutex.h */
52 #include <net/netisr.h>
53 #include <net/vnet.h>
54 
55 #include <netinet/in.h>
56 #include <netinet/ip.h>		/* ip_len, ip_off */
57 #include <netinet/ip_var.h>	/* ip_output(), IP_FORWARDING */
58 #include <netinet/ip_fw.h>
59 #include <netinet/ip_dummynet.h>
60 #include <netinet/if_ether.h> /* various ether_* routines */
61 #include <netinet/ip6.h>       /* for ip6_input, ip6_output prototypes */
62 #include <netinet6/ip6_var.h>
63 #include <netpfil/ipfw/dn_heap.h>
64 
65 #ifdef NEW_AQM
66 #include <netpfil/ipfw/ip_fw_private.h>
67 #include <netpfil/ipfw/ip_dn_private.h>
68 #include <netpfil/ipfw/dn_aqm.h>
69 #include <netpfil/ipfw/dn_aqm_codel.h>
70 #include <netpfil/ipfw/dn_sched.h>
71 
72 #define DN_AQM_CODEL 1
73 
74 static struct dn_aqm codel_desc;
75 
76 /* default codel parameters */
77 struct dn_aqm_codel_parms codel_sysctl = {5000 * AQM_TIME_1US,
78 	100000 * AQM_TIME_1US, 0};
79 
80 static int
81 codel_sysctl_interval_handler(SYSCTL_HANDLER_ARGS)
82 {
83 	int error;
84 	long  value;
85 
86 	value = codel_sysctl.interval;
87 	value /= AQM_TIME_1US;
88 	error = sysctl_handle_long(oidp, &value, 0, req);
89 	if (error != 0 || req->newptr == NULL)
90 		return (error);
91 	if (value < 1 || value > 100 * AQM_TIME_1S)
92 		return (EINVAL);
93 	codel_sysctl.interval = value * AQM_TIME_1US ;
94 	return (0);
95 }
96 
97 static int
98 codel_sysctl_target_handler(SYSCTL_HANDLER_ARGS)
99 {
100 	int error;
101 	long  value;
102 
103 	value = codel_sysctl.target;
104 	value /= AQM_TIME_1US;
105 	error = sysctl_handle_long(oidp, &value, 0, req);
106 	if (error != 0 || req->newptr == NULL)
107 		return (error);
108 	D("%ld", value);
109 	if (value < 1 || value > 5 * AQM_TIME_1S)
110 		return (EINVAL);
111 	codel_sysctl.target = value * AQM_TIME_1US ;
112 	return (0);
113 }
114 
115 /* defining Codel sysctl variables */
116 SYSBEGIN(f4)
117 
118 SYSCTL_DECL(_net_inet);
119 SYSCTL_DECL(_net_inet_ip);
120 SYSCTL_DECL(_net_inet_ip_dummynet);
121 static SYSCTL_NODE(_net_inet_ip_dummynet, OID_AUTO, codel,
122     CTLFLAG_RW | CTLFLAG_MPSAFE, 0,
123     "CODEL");
124 
125 #ifdef SYSCTL_NODE
126 SYSCTL_PROC(_net_inet_ip_dummynet_codel, OID_AUTO, target,
127     CTLTYPE_LONG | CTLFLAG_RW | CTLFLAG_NEEDGIANT,
128     NULL, 0,codel_sysctl_target_handler, "L",
129     "CoDel target in microsecond");
130 
131 SYSCTL_PROC(_net_inet_ip_dummynet_codel, OID_AUTO, interval,
132     CTLTYPE_LONG | CTLFLAG_RW | CTLFLAG_NEEDGIANT,
133     NULL, 0, codel_sysctl_interval_handler, "L",
134     "CoDel interval in microsecond");
135 #endif
136 
137 /* This function computes codel_interval/sqrt(count)
138  *  Newton's method of approximation is used to compute 1/sqrt(count).
139  * http://betterexplained.com/articles/
140  * 	understanding-quakes-fast-inverse-square-root/
141  */
142 aqm_time_t
143 control_law(struct codel_status *cst, struct dn_aqm_codel_parms *cprms,
144 	aqm_time_t t)
145 {
146 	uint32_t count;
147 	uint64_t temp;
148 	count = cst->count;
149 
150 	/* we don't calculate isqrt(1) to get more accurate result*/
151 	if (count == 1) {
152 		/* prepare isqrt (old guess) for the next iteration i.e. 1/sqrt(2)*/
153 		cst->isqrt = (1UL<< FIX_POINT_BITS) * 7/10;
154 		/* return time + isqrt(1)*interval */
155 		return t + cprms->interval;
156 	}
157 
158 	/* newguess = g(1.5 - 0.5*c*g^2)
159 	 * Multiplying both sides by 2 to make all the constants integers
160 	 * newguess * 2  = g(3 - c*g^2) g=old guess, c=count
161 	 * So, newguess = newguess /2
162 	 * Fixed point operations are used here.
163 	 */
164 
165 	/* Calculate g^2 */
166 	temp = (uint32_t) cst->isqrt * cst->isqrt;
167 	/* Calculate (3 - c*g^2) i.e. (3 - c * temp) */
168 	temp = (3ULL<< (FIX_POINT_BITS*2)) - (count * temp);
169 
170 	/*
171 	 * Divide by 2 because we multiplied the original equation by two
172 	 * Also, we shift the result by 8 bits to prevent overflow.
173 	 * */
174 	temp >>= (1 + 8);
175 
176 	/*  Now, temp = (1.5 - 0.5*c*g^2)
177 	 * Calculate g (1.5 - 0.5*c*g^2) i.e. g * temp
178 	 */
179 	temp = (cst->isqrt * temp) >> (FIX_POINT_BITS + FIX_POINT_BITS - 8);
180 	cst->isqrt = temp;
181 
182 	 /* calculate codel_interval/sqrt(count) */
183 	 return t + ((cprms->interval * temp) >> FIX_POINT_BITS);
184 }
185 
186 /*
187  * Extract a packet from the head of queue 'q'
188  * Return a packet or NULL if the queue is empty.
189  * Also extract packet's timestamp from mtag.
190  */
191 struct mbuf *
192 codel_extract_head(struct dn_queue *q, aqm_time_t *pkt_ts)
193 {
194 	struct m_tag *mtag;
195 	struct mbuf *m;
196 
197 next:	m = q->mq.head;
198 	if (m == NULL)
199 		return m;
200 	q->mq.head = m->m_nextpkt;
201 
202 	/* Update stats */
203 	update_stats(q, -m->m_pkthdr.len, 0);
204 
205 	if (q->ni.length == 0) /* queue is now idle */
206 			q->q_time = V_dn_cfg.curr_time;
207 
208 	/* extract packet TS*/
209 	mtag = m_tag_locate(m, MTAG_ABI_COMPAT, DN_AQM_MTAG_TS, NULL);
210 	if (mtag == NULL) {
211 		D("Codel timestamp mtag not found!");
212 		*pkt_ts = 0;
213 	} else {
214 		*pkt_ts = *(aqm_time_t *)(mtag + 1);
215 		m_tag_delete(m,mtag);
216 	}
217 	if (m->m_pkthdr.rcvif != NULL &&
218 	    __predict_false(m_rcvif_restore(m) == NULL)) {
219 		m_freem(m);
220 		goto next;
221 	}
222 
223 	return m;
224 }
225 
226 /*
227  * Enqueue a packet 'm' in queue 'q'
228  */
229 static int
230 aqm_codel_enqueue(struct dn_queue *q, struct mbuf *m)
231 {
232 	struct dn_fs *f;
233 	uint64_t len;
234 	struct codel_status *cst;	/*codel status variables */
235 	struct m_tag *mtag;
236 
237 	f = &(q->fs->fs);
238 	len = m->m_pkthdr.len;
239 	cst = q->aqm_status;
240 	if(!cst) {
241 		D("Codel queue is not initialized\n");
242 		goto drop;
243 	}
244 
245 	/* Finding maximum packet size */
246 	// XXX we can get MTU from driver instead
247 	if (len > cst->maxpkt_size)
248 		cst->maxpkt_size = len;
249 
250 	/* check for queue size and drop the tail if exceed queue limit*/
251 	if (f->flags & DN_QSIZE_BYTES) {
252 		if ( q->ni.len_bytes > f->qsize)
253 			goto drop;
254 	}
255 	else {
256 		if ( q->ni.length >= f->qsize)
257 			goto drop;
258 	}
259 
260 	/* Add timestamp as mtag */
261 	mtag = m_tag_locate(m, MTAG_ABI_COMPAT, DN_AQM_MTAG_TS, NULL);
262 	if (mtag == NULL)
263 		mtag = m_tag_alloc(MTAG_ABI_COMPAT, DN_AQM_MTAG_TS,
264 			sizeof(aqm_time_t), M_NOWAIT);
265 	if (mtag == NULL)
266 		goto drop;
267 
268 	*(aqm_time_t *)(mtag + 1) = AQM_UNOW;
269 	m_tag_prepend(m, mtag);
270 
271 	mq_append(&q->mq, m);
272 	update_stats(q, len, 0);
273 	return (0);
274 
275 drop:
276 	update_stats(q, 0, 1);
277 	FREE_PKT(m);
278 	return (1);
279 }
280 
281 /* Dequeue a pcaket from queue q */
282 static struct mbuf *
283 aqm_codel_dequeue(struct dn_queue *q)
284 {
285 	return codel_dequeue(q);
286 }
287 
288 /*
289  * initialize Codel for queue 'q'
290  * First allocate memory for codel status.
291  */
292 static int
293 aqm_codel_init(struct dn_queue *q)
294 {
295 	struct codel_status *cst;
296 
297 	if (!q->fs->aqmcfg) {
298 		D("Codel is not configure!d");
299 		return EINVAL;
300 	}
301 
302 	q->aqm_status = malloc(sizeof(struct codel_status),
303 			 M_DUMMYNET, M_NOWAIT | M_ZERO);
304 	if (q->aqm_status == NULL) {
305 		D("Cannot allocate AQM_codel private data");
306 		return ENOMEM ;
307 	}
308 
309 	/* init codel status variables */
310 	cst = q->aqm_status;
311 	cst->dropping=0;
312 	cst->first_above_time=0;
313 	cst->drop_next_time=0;
314 	cst->count=0;
315 	cst->maxpkt_size = 500;
316 
317 	/* increase reference counters */
318 	codel_desc.ref_count++;
319 
320 	return 0;
321 }
322 
323 /*
324  * Clean up Codel status for queue 'q'
325  * Destroy memory allocated for codel status.
326  */
327 static int
328 aqm_codel_cleanup(struct dn_queue *q)
329 {
330 
331 	if (q && q->aqm_status) {
332 		free(q->aqm_status, M_DUMMYNET);
333 		q->aqm_status = NULL;
334 		/* decrease reference counters */
335 		codel_desc.ref_count--;
336 	}
337 	else
338 		D("Codel already cleaned up");
339 	return 0;
340 }
341 
342 /*
343  * Config codel parameters
344  * also allocate memory for codel configurations
345  */
346 static int
347 aqm_codel_config(struct dn_fsk* fs, struct dn_extra_parms *ep, int len)
348 {
349 	struct dn_aqm_codel_parms *ccfg;
350 
351 	int l = sizeof(struct dn_extra_parms);
352 	if (len < l) {
353 		D("invalid sched parms length got %d need %d", len, l);
354 		return EINVAL;
355 	}
356 	/* we free the old cfg because maybe the original allocation
357 	 * not the same size as the new one (different AQM type).
358 	 */
359 	if (fs->aqmcfg) {
360 		free(fs->aqmcfg, M_DUMMYNET);
361 		fs->aqmcfg = NULL;
362 	}
363 
364 	fs->aqmcfg = malloc(sizeof(struct dn_aqm_codel_parms),
365 			 M_DUMMYNET, M_NOWAIT | M_ZERO);
366 	if (fs->aqmcfg== NULL) {
367 		D("cannot allocate AQM_codel configuration parameters");
368 		return ENOMEM;
369 	}
370 
371 	/* configure codel parameters */
372 	ccfg = fs->aqmcfg;
373 
374 	if (ep->par[0] < 0)
375 		ccfg->target = codel_sysctl.target;
376 	else
377 		ccfg->target = ep->par[0] * AQM_TIME_1US;
378 
379 	if (ep->par[1] < 0)
380 		ccfg->interval = codel_sysctl.interval;
381 	else
382 		ccfg->interval = ep->par[1] * AQM_TIME_1US;
383 
384 	if (ep->par[2] < 0)
385 		ccfg->flags = 0;
386 	else
387 		ccfg->flags = ep->par[2];
388 
389 	/* bound codel configurations */
390 	ccfg->target = BOUND_VAR(ccfg->target,1, 5 * AQM_TIME_1S);
391 	ccfg->interval = BOUND_VAR(ccfg->interval,1, 5 * AQM_TIME_1S);
392 	/* increase config reference counter */
393 	codel_desc.cfg_ref_count++;
394 
395 	return 0;
396 }
397 
398 /*
399  * Deconfigure Codel and free memory allocation
400  */
401 static int
402 aqm_codel_deconfig(struct dn_fsk* fs)
403 {
404 
405 	if (fs && fs->aqmcfg) {
406 		free(fs->aqmcfg, M_DUMMYNET);
407 		fs->aqmcfg = NULL;
408 		fs->aqmfp = NULL;
409 		/* decrease config reference counter */
410 		codel_desc.cfg_ref_count--;
411 	}
412 
413 	return 0;
414 }
415 
416 /*
417  * Retrieve Codel configuration parameters.
418  */
419 static int
420 aqm_codel_getconfig(struct dn_fsk *fs, struct dn_extra_parms * ep)
421 {
422 	struct dn_aqm_codel_parms *ccfg;
423 
424 	if (fs->aqmcfg) {
425 		strlcpy(ep->name, codel_desc.name, sizeof(ep->name));
426 		ccfg = fs->aqmcfg;
427 		ep->par[0] = ccfg->target / AQM_TIME_1US;
428 		ep->par[1] = ccfg->interval / AQM_TIME_1US;
429 		ep->par[2] = ccfg->flags;
430 		return 0;
431 	}
432 	return 1;
433 }
434 
435 static struct dn_aqm codel_desc = {
436 	_SI( .type = )  DN_AQM_CODEL,
437 	_SI( .name = )  "CODEL",
438 	_SI( .enqueue = )  aqm_codel_enqueue,
439 	_SI( .dequeue = )  aqm_codel_dequeue,
440 	_SI( .config = )  aqm_codel_config,
441 	_SI( .getconfig = )  aqm_codel_getconfig,
442 	_SI( .deconfig = )  aqm_codel_deconfig,
443 	_SI( .init = )  aqm_codel_init,
444 	_SI( .cleanup = )  aqm_codel_cleanup,
445 };
446 
447 DECLARE_DNAQM_MODULE(dn_aqm_codel, &codel_desc);
448 
449 #endif
450