1c97bf8c3SAlexander Motin /*-
24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
3fe267a55SPedro F. Giffuni *
4c97bf8c3SAlexander Motin * Copyright (c) 2005 Nuno Antunes <nuno.antunes@gmail.com>
5c97bf8c3SAlexander Motin * Copyright (c) 2007 Alexander Motin <mav@freebsd.org>
6d0d2e523SLutz Donnerhacke * Copyright (c) 2019 Lutz Donnerhacke <lutz@donnerhacke.de>
7c97bf8c3SAlexander Motin * All rights reserved.
8c97bf8c3SAlexander Motin *
9c97bf8c3SAlexander Motin * Redistribution and use in source and binary forms, with or without
10c97bf8c3SAlexander Motin * modification, are permitted provided that the following conditions
11c97bf8c3SAlexander Motin * are met:
12c97bf8c3SAlexander Motin * 1. Redistributions of source code must retain the above copyright
13c97bf8c3SAlexander Motin * notice, this list of conditions and the following disclaimer.
14c97bf8c3SAlexander Motin * 2. Redistributions in binary form must reproduce the above copyright
15c97bf8c3SAlexander Motin * notice, this list of conditions and the following disclaimer in the
16c97bf8c3SAlexander Motin * documentation and/or other materials provided with the distribution.
17c97bf8c3SAlexander Motin *
18c97bf8c3SAlexander Motin * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19c97bf8c3SAlexander Motin * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20c97bf8c3SAlexander Motin * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21c97bf8c3SAlexander Motin * ARE DISCLAIMED. IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
22c97bf8c3SAlexander Motin * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23c97bf8c3SAlexander Motin * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24c97bf8c3SAlexander Motin * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25c97bf8c3SAlexander Motin * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26c97bf8c3SAlexander Motin * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27c97bf8c3SAlexander Motin * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28c97bf8c3SAlexander Motin * SUCH DAMAGE.
29c97bf8c3SAlexander Motin */
30c97bf8c3SAlexander Motin
31c97bf8c3SAlexander Motin /*
32053359b7SPedro F. Giffuni * ng_car - An implementation of committed access rate for netgraph
33c97bf8c3SAlexander Motin *
34c97bf8c3SAlexander Motin * TODO:
35c97bf8c3SAlexander Motin * - Sanitize input config values (impose some limits)
36c97bf8c3SAlexander Motin * - Implement DSCP marking for IPv4
37d0d2e523SLutz Donnerhacke * - Decouple functionality into a simple classifier (g/y/r)
38d0d2e523SLutz Donnerhacke * and various action nodes (i.e. shape, dcsp, pcp)
39c97bf8c3SAlexander Motin */
40c97bf8c3SAlexander Motin
41c97bf8c3SAlexander Motin #include <sys/param.h>
42c97bf8c3SAlexander Motin #include <sys/errno.h>
43ae1be01fSAlexander Motin #include <sys/kernel.h>
44ae1be01fSAlexander Motin #include <sys/malloc.h>
45ae1be01fSAlexander Motin #include <sys/mbuf.h>
46c97bf8c3SAlexander Motin
47c97bf8c3SAlexander Motin #include <netgraph/ng_message.h>
48c97bf8c3SAlexander Motin #include <netgraph/ng_parse.h>
49c97bf8c3SAlexander Motin #include <netgraph/netgraph.h>
50c97bf8c3SAlexander Motin #include <netgraph/ng_car.h>
51c97bf8c3SAlexander Motin
52d0d2e523SLutz Donnerhacke #include "qos.h"
53d0d2e523SLutz Donnerhacke
54*9b8db664SDmitry Lukhtionov #ifdef NG_SEPARATE_MALLOC
55*9b8db664SDmitry Lukhtionov static MALLOC_DEFINE(M_NETGRAPH_CAR, "netgraph_car", "netgraph car node");
56*9b8db664SDmitry Lukhtionov #else
57*9b8db664SDmitry Lukhtionov #define M_NETGRAPH_CAR M_NETGRAPH
58*9b8db664SDmitry Lukhtionov #endif
59*9b8db664SDmitry Lukhtionov
60c97bf8c3SAlexander Motin #define NG_CAR_QUEUE_SIZE 100 /* Maximum queue size for SHAPE mode */
61053359b7SPedro F. Giffuni #define NG_CAR_QUEUE_MIN_TH 8 /* Minimum RED threshold for SHAPE mode */
62c97bf8c3SAlexander Motin
63c97bf8c3SAlexander Motin /* Hook private info */
64c97bf8c3SAlexander Motin struct hookinfo {
65c97bf8c3SAlexander Motin hook_p hook; /* this (source) hook */
66c97bf8c3SAlexander Motin hook_p dest; /* destination hook */
67c97bf8c3SAlexander Motin
68053359b7SPedro F. Giffuni int64_t tc; /* committed token bucket counter */
69c97bf8c3SAlexander Motin int64_t te; /* exceeded/peak token bucket counter */
70c86d865eSAlexander Motin struct bintime lastRefill; /* last token refill time */
71c97bf8c3SAlexander Motin
72c97bf8c3SAlexander Motin struct ng_car_hookconf conf; /* hook configuration */
73c97bf8c3SAlexander Motin struct ng_car_hookstats stats; /* hook stats */
74c97bf8c3SAlexander Motin
75c97bf8c3SAlexander Motin struct mbuf *q[NG_CAR_QUEUE_SIZE]; /* circular packet queue */
76c77b232bSAlexander Motin u_int q_first; /* first queue element */
77c77b232bSAlexander Motin u_int q_last; /* last queue element */
78c97bf8c3SAlexander Motin struct callout q_callout; /* periodic queue processing routine */
79c97bf8c3SAlexander Motin struct mtx q_mtx; /* queue mutex */
80c97bf8c3SAlexander Motin };
81c97bf8c3SAlexander Motin
82c97bf8c3SAlexander Motin /* Private information for each node instance */
83c97bf8c3SAlexander Motin struct privdata {
84c97bf8c3SAlexander Motin node_p node; /* the node itself */
85c97bf8c3SAlexander Motin struct hookinfo upper; /* hook to upper layers */
86c97bf8c3SAlexander Motin struct hookinfo lower; /* hook to lower layers */
87c97bf8c3SAlexander Motin };
88c97bf8c3SAlexander Motin typedef struct privdata *priv_p;
89c97bf8c3SAlexander Motin
90c97bf8c3SAlexander Motin static ng_constructor_t ng_car_constructor;
91c97bf8c3SAlexander Motin static ng_rcvmsg_t ng_car_rcvmsg;
92c97bf8c3SAlexander Motin static ng_shutdown_t ng_car_shutdown;
93c97bf8c3SAlexander Motin static ng_newhook_t ng_car_newhook;
94c97bf8c3SAlexander Motin static ng_rcvdata_t ng_car_rcvdata;
95c97bf8c3SAlexander Motin static ng_disconnect_t ng_car_disconnect;
96c97bf8c3SAlexander Motin
97c97bf8c3SAlexander Motin static void ng_car_refillhook(struct hookinfo *h);
98c97bf8c3SAlexander Motin static void ng_car_schedule(struct hookinfo *h);
99c97bf8c3SAlexander Motin void ng_car_q_event(node_p node, hook_p hook, void *arg, int arg2);
100c97bf8c3SAlexander Motin static void ng_car_enqueue(struct hookinfo *h, item_p item);
101c97bf8c3SAlexander Motin
102c97bf8c3SAlexander Motin /* Parse type for struct ng_car_hookstats */
103c97bf8c3SAlexander Motin static const struct ng_parse_struct_field ng_car_hookstats_type_fields[]
104c97bf8c3SAlexander Motin = NG_CAR_HOOKSTATS;
105c97bf8c3SAlexander Motin static const struct ng_parse_type ng_car_hookstats_type = {
106c97bf8c3SAlexander Motin &ng_parse_struct_type,
107c97bf8c3SAlexander Motin &ng_car_hookstats_type_fields
108c97bf8c3SAlexander Motin };
109c97bf8c3SAlexander Motin
110c97bf8c3SAlexander Motin /* Parse type for struct ng_car_bulkstats */
111c97bf8c3SAlexander Motin static const struct ng_parse_struct_field ng_car_bulkstats_type_fields[]
112c97bf8c3SAlexander Motin = NG_CAR_BULKSTATS(&ng_car_hookstats_type);
113c97bf8c3SAlexander Motin static const struct ng_parse_type ng_car_bulkstats_type = {
114c97bf8c3SAlexander Motin &ng_parse_struct_type,
115c97bf8c3SAlexander Motin &ng_car_bulkstats_type_fields
116c97bf8c3SAlexander Motin };
117c97bf8c3SAlexander Motin
118c97bf8c3SAlexander Motin /* Parse type for struct ng_car_hookconf */
119c97bf8c3SAlexander Motin static const struct ng_parse_struct_field ng_car_hookconf_type_fields[]
120c97bf8c3SAlexander Motin = NG_CAR_HOOKCONF;
121c97bf8c3SAlexander Motin static const struct ng_parse_type ng_car_hookconf_type = {
122c97bf8c3SAlexander Motin &ng_parse_struct_type,
123c97bf8c3SAlexander Motin &ng_car_hookconf_type_fields
124c97bf8c3SAlexander Motin };
125c97bf8c3SAlexander Motin
126c97bf8c3SAlexander Motin /* Parse type for struct ng_car_bulkconf */
127c97bf8c3SAlexander Motin static const struct ng_parse_struct_field ng_car_bulkconf_type_fields[]
128c97bf8c3SAlexander Motin = NG_CAR_BULKCONF(&ng_car_hookconf_type);
129c97bf8c3SAlexander Motin static const struct ng_parse_type ng_car_bulkconf_type = {
130c97bf8c3SAlexander Motin &ng_parse_struct_type,
131c97bf8c3SAlexander Motin &ng_car_bulkconf_type_fields
132c97bf8c3SAlexander Motin };
133c97bf8c3SAlexander Motin
134c97bf8c3SAlexander Motin /* Command list */
135c97bf8c3SAlexander Motin static struct ng_cmdlist ng_car_cmdlist[] = {
136c97bf8c3SAlexander Motin {
137c97bf8c3SAlexander Motin NGM_CAR_COOKIE,
138c97bf8c3SAlexander Motin NGM_CAR_GET_STATS,
139c97bf8c3SAlexander Motin "getstats",
140c97bf8c3SAlexander Motin NULL,
141c97bf8c3SAlexander Motin &ng_car_bulkstats_type,
142c97bf8c3SAlexander Motin },
143c97bf8c3SAlexander Motin {
144c97bf8c3SAlexander Motin NGM_CAR_COOKIE,
145c97bf8c3SAlexander Motin NGM_CAR_CLR_STATS,
146c97bf8c3SAlexander Motin "clrstats",
147c97bf8c3SAlexander Motin NULL,
148c97bf8c3SAlexander Motin NULL,
149c97bf8c3SAlexander Motin },
150c97bf8c3SAlexander Motin {
151c97bf8c3SAlexander Motin NGM_CAR_COOKIE,
152c97bf8c3SAlexander Motin NGM_CAR_GETCLR_STATS,
153c97bf8c3SAlexander Motin "getclrstats",
154c97bf8c3SAlexander Motin NULL,
155c97bf8c3SAlexander Motin &ng_car_bulkstats_type,
156c97bf8c3SAlexander Motin },
157c97bf8c3SAlexander Motin
158c97bf8c3SAlexander Motin {
159c97bf8c3SAlexander Motin NGM_CAR_COOKIE,
160c97bf8c3SAlexander Motin NGM_CAR_GET_CONF,
161c97bf8c3SAlexander Motin "getconf",
162c97bf8c3SAlexander Motin NULL,
163c97bf8c3SAlexander Motin &ng_car_bulkconf_type,
164c97bf8c3SAlexander Motin },
165c97bf8c3SAlexander Motin {
166c97bf8c3SAlexander Motin NGM_CAR_COOKIE,
167c97bf8c3SAlexander Motin NGM_CAR_SET_CONF,
168c97bf8c3SAlexander Motin "setconf",
169c97bf8c3SAlexander Motin &ng_car_bulkconf_type,
170c97bf8c3SAlexander Motin NULL,
171c97bf8c3SAlexander Motin },
172c97bf8c3SAlexander Motin { 0 }
173c97bf8c3SAlexander Motin };
174c97bf8c3SAlexander Motin
175c97bf8c3SAlexander Motin /* Netgraph node type descriptor */
176c97bf8c3SAlexander Motin static struct ng_type ng_car_typestruct = {
177c97bf8c3SAlexander Motin .version = NG_ABI_VERSION,
178c97bf8c3SAlexander Motin .name = NG_CAR_NODE_TYPE,
179c97bf8c3SAlexander Motin .constructor = ng_car_constructor,
180c97bf8c3SAlexander Motin .rcvmsg = ng_car_rcvmsg,
181c97bf8c3SAlexander Motin .shutdown = ng_car_shutdown,
182c97bf8c3SAlexander Motin .newhook = ng_car_newhook,
183c97bf8c3SAlexander Motin .rcvdata = ng_car_rcvdata,
184c97bf8c3SAlexander Motin .disconnect = ng_car_disconnect,
185c97bf8c3SAlexander Motin .cmdlist = ng_car_cmdlist,
186c97bf8c3SAlexander Motin };
187c97bf8c3SAlexander Motin NETGRAPH_INIT(car, &ng_car_typestruct);
188c97bf8c3SAlexander Motin
189c97bf8c3SAlexander Motin /*
190c97bf8c3SAlexander Motin * Node constructor
191c97bf8c3SAlexander Motin */
192c97bf8c3SAlexander Motin static int
ng_car_constructor(node_p node)193c97bf8c3SAlexander Motin ng_car_constructor(node_p node)
194c97bf8c3SAlexander Motin {
195c97bf8c3SAlexander Motin priv_p priv;
196c97bf8c3SAlexander Motin
197ae1be01fSAlexander Motin /* Initialize private descriptor. */
198*9b8db664SDmitry Lukhtionov priv = malloc(sizeof(*priv), M_NETGRAPH_CAR, M_WAITOK | M_ZERO);
199c97bf8c3SAlexander Motin
200c97bf8c3SAlexander Motin NG_NODE_SET_PRIVATE(node, priv);
201c97bf8c3SAlexander Motin priv->node = node;
202c97bf8c3SAlexander Motin
203c97bf8c3SAlexander Motin /*
204c97bf8c3SAlexander Motin * Arbitrary default values
205c97bf8c3SAlexander Motin */
206c97bf8c3SAlexander Motin
207c97bf8c3SAlexander Motin priv->upper.hook = NULL;
208c97bf8c3SAlexander Motin priv->upper.dest = NULL;
209c97bf8c3SAlexander Motin priv->upper.tc = priv->upper.conf.cbs = NG_CAR_CBS_MIN;
210c97bf8c3SAlexander Motin priv->upper.te = priv->upper.conf.ebs = NG_CAR_EBS_MIN;
211c97bf8c3SAlexander Motin priv->upper.conf.cir = NG_CAR_CIR_DFLT;
212c97bf8c3SAlexander Motin priv->upper.conf.green_action = NG_CAR_ACTION_FORWARD;
213c97bf8c3SAlexander Motin priv->upper.conf.yellow_action = NG_CAR_ACTION_FORWARD;
214c97bf8c3SAlexander Motin priv->upper.conf.red_action = NG_CAR_ACTION_DROP;
215c97bf8c3SAlexander Motin priv->upper.conf.mode = 0;
216c86d865eSAlexander Motin getbinuptime(&priv->upper.lastRefill);
217c97bf8c3SAlexander Motin priv->upper.q_first = 0;
218c97bf8c3SAlexander Motin priv->upper.q_last = 0;
219c97bf8c3SAlexander Motin ng_callout_init(&priv->upper.q_callout);
220c97bf8c3SAlexander Motin mtx_init(&priv->upper.q_mtx, "ng_car_u", NULL, MTX_DEF);
221c97bf8c3SAlexander Motin
222c97bf8c3SAlexander Motin priv->lower.hook = NULL;
223c97bf8c3SAlexander Motin priv->lower.dest = NULL;
224c97bf8c3SAlexander Motin priv->lower.tc = priv->lower.conf.cbs = NG_CAR_CBS_MIN;
225c97bf8c3SAlexander Motin priv->lower.te = priv->lower.conf.ebs = NG_CAR_EBS_MIN;
226c97bf8c3SAlexander Motin priv->lower.conf.cir = NG_CAR_CIR_DFLT;
227c97bf8c3SAlexander Motin priv->lower.conf.green_action = NG_CAR_ACTION_FORWARD;
228c97bf8c3SAlexander Motin priv->lower.conf.yellow_action = NG_CAR_ACTION_FORWARD;
229c97bf8c3SAlexander Motin priv->lower.conf.red_action = NG_CAR_ACTION_DROP;
230c97bf8c3SAlexander Motin priv->lower.conf.mode = 0;
231c97bf8c3SAlexander Motin priv->lower.lastRefill = priv->upper.lastRefill;
232c97bf8c3SAlexander Motin priv->lower.q_first = 0;
233c97bf8c3SAlexander Motin priv->lower.q_last = 0;
234c97bf8c3SAlexander Motin ng_callout_init(&priv->lower.q_callout);
235c97bf8c3SAlexander Motin mtx_init(&priv->lower.q_mtx, "ng_car_l", NULL, MTX_DEF);
236c97bf8c3SAlexander Motin
237c97bf8c3SAlexander Motin return (0);
238c97bf8c3SAlexander Motin }
239c97bf8c3SAlexander Motin
240c97bf8c3SAlexander Motin /*
241ae1be01fSAlexander Motin * Add a hook.
242c97bf8c3SAlexander Motin */
243c97bf8c3SAlexander Motin static int
ng_car_newhook(node_p node,hook_p hook,const char * name)244c97bf8c3SAlexander Motin ng_car_newhook(node_p node, hook_p hook, const char *name)
245c97bf8c3SAlexander Motin {
246c97bf8c3SAlexander Motin const priv_p priv = NG_NODE_PRIVATE(node);
247c97bf8c3SAlexander Motin
248c97bf8c3SAlexander Motin if (strcmp(name, NG_CAR_HOOK_LOWER) == 0) {
249c97bf8c3SAlexander Motin priv->lower.hook = hook;
250c97bf8c3SAlexander Motin priv->upper.dest = hook;
251c97bf8c3SAlexander Motin bzero(&priv->lower.stats, sizeof(priv->lower.stats));
252c97bf8c3SAlexander Motin NG_HOOK_SET_PRIVATE(hook, &priv->lower);
253c97bf8c3SAlexander Motin } else if (strcmp(name, NG_CAR_HOOK_UPPER) == 0) {
254c97bf8c3SAlexander Motin priv->upper.hook = hook;
255c97bf8c3SAlexander Motin priv->lower.dest = hook;
256c97bf8c3SAlexander Motin bzero(&priv->upper.stats, sizeof(priv->upper.stats));
257c97bf8c3SAlexander Motin NG_HOOK_SET_PRIVATE(hook, &priv->upper);
258c97bf8c3SAlexander Motin } else
259c97bf8c3SAlexander Motin return (EINVAL);
260c97bf8c3SAlexander Motin return(0);
261c97bf8c3SAlexander Motin }
262c97bf8c3SAlexander Motin
263c97bf8c3SAlexander Motin /*
264ae1be01fSAlexander Motin * Data has arrived.
265c97bf8c3SAlexander Motin */
266c97bf8c3SAlexander Motin static int
ng_car_rcvdata(hook_p hook,item_p item)267c97bf8c3SAlexander Motin ng_car_rcvdata(hook_p hook, item_p item )
268c97bf8c3SAlexander Motin {
269c97bf8c3SAlexander Motin struct hookinfo *const hinfo = NG_HOOK_PRIVATE(hook);
270c77b232bSAlexander Motin struct mbuf *m;
271d0d2e523SLutz Donnerhacke struct m_qos_color *colp;
272d0d2e523SLutz Donnerhacke enum qos_color col;
273c97bf8c3SAlexander Motin int error = 0;
274c77b232bSAlexander Motin u_int len;
275c97bf8c3SAlexander Motin
276c97bf8c3SAlexander Motin /* If queue is not empty now then enqueue packet. */
277c97bf8c3SAlexander Motin if (hinfo->q_first != hinfo->q_last) {
278c97bf8c3SAlexander Motin ng_car_enqueue(hinfo, item);
279c97bf8c3SAlexander Motin return (0);
280c97bf8c3SAlexander Motin }
281c97bf8c3SAlexander Motin
282c97bf8c3SAlexander Motin m = NGI_M(item);
283c97bf8c3SAlexander Motin
284d0d2e523SLutz Donnerhacke #define NG_CAR_PERFORM_MATCH_ACTION(a,col) \
285c97bf8c3SAlexander Motin do { \
286c97bf8c3SAlexander Motin switch (a) { \
287c97bf8c3SAlexander Motin case NG_CAR_ACTION_FORWARD: \
288ae1be01fSAlexander Motin /* Do nothing. */ \
289c97bf8c3SAlexander Motin break; \
290c97bf8c3SAlexander Motin case NG_CAR_ACTION_MARK: \
291d0d2e523SLutz Donnerhacke if (colp == NULL) { \
292d0d2e523SLutz Donnerhacke colp = (void *)m_tag_alloc( \
293d0d2e523SLutz Donnerhacke M_QOS_COOKIE, M_QOS_COLOR, \
294d0d2e523SLutz Donnerhacke MTAG_SIZE(m_qos_color), M_NOWAIT); \
295d0d2e523SLutz Donnerhacke if (colp != NULL) \
296d0d2e523SLutz Donnerhacke m_tag_prepend(m, &colp->tag); \
297d0d2e523SLutz Donnerhacke } \
298d0d2e523SLutz Donnerhacke if (colp != NULL) \
299d0d2e523SLutz Donnerhacke colp->color = col; \
300c97bf8c3SAlexander Motin break; \
301c97bf8c3SAlexander Motin case NG_CAR_ACTION_DROP: \
302c97bf8c3SAlexander Motin default: \
303ae1be01fSAlexander Motin /* Drop packet and return. */ \
304c97bf8c3SAlexander Motin NG_FREE_ITEM(item); \
305def4e701SGleb Smirnoff ++hinfo->stats.dropped_pkts; \
306ae1be01fSAlexander Motin return (0); \
307c97bf8c3SAlexander Motin } \
308ae1be01fSAlexander Motin } while (0)
309c97bf8c3SAlexander Motin
310673f5a8bSAlexander Motin /* Packet is counted as 128 tokens for better resolution */
311673f5a8bSAlexander Motin if (hinfo->conf.opt & NG_CAR_COUNT_PACKETS) {
312673f5a8bSAlexander Motin len = 128;
313673f5a8bSAlexander Motin } else {
314673f5a8bSAlexander Motin len = m->m_pkthdr.len;
315673f5a8bSAlexander Motin }
316673f5a8bSAlexander Motin
317d0d2e523SLutz Donnerhacke /* Determine current color of the packet (default green) */
318d0d2e523SLutz Donnerhacke colp = (void *)m_tag_locate(m, M_QOS_COOKIE, M_QOS_COLOR, NULL);
319d0d2e523SLutz Donnerhacke if ((hinfo->conf.opt & NG_CAR_COLOR_AWARE) && (colp != NULL))
320d0d2e523SLutz Donnerhacke col = colp->color;
321d0d2e523SLutz Donnerhacke else
322d0d2e523SLutz Donnerhacke col = QOS_COLOR_GREEN;
323d0d2e523SLutz Donnerhacke
324053359b7SPedro F. Giffuni /* Check committed token bucket. */
325d0d2e523SLutz Donnerhacke if (hinfo->tc - len >= 0 && col <= QOS_COLOR_GREEN) {
326ae1be01fSAlexander Motin /* This packet is green. */
327c97bf8c3SAlexander Motin ++hinfo->stats.green_pkts;
328673f5a8bSAlexander Motin hinfo->tc -= len;
329d0d2e523SLutz Donnerhacke NG_CAR_PERFORM_MATCH_ACTION(
330d0d2e523SLutz Donnerhacke hinfo->conf.green_action,
331d0d2e523SLutz Donnerhacke QOS_COLOR_GREEN);
332c97bf8c3SAlexander Motin } else {
333c97bf8c3SAlexander Motin /* Refill only if not green without it. */
334c97bf8c3SAlexander Motin ng_car_refillhook(hinfo);
335c97bf8c3SAlexander Motin
336053359b7SPedro F. Giffuni /* Check committed token bucket again after refill. */
337d0d2e523SLutz Donnerhacke if (hinfo->tc - len >= 0 && col <= QOS_COLOR_GREEN) {
338c97bf8c3SAlexander Motin /* This packet is green */
339c97bf8c3SAlexander Motin ++hinfo->stats.green_pkts;
340673f5a8bSAlexander Motin hinfo->tc -= len;
341d0d2e523SLutz Donnerhacke NG_CAR_PERFORM_MATCH_ACTION(
342d0d2e523SLutz Donnerhacke hinfo->conf.green_action,
343d0d2e523SLutz Donnerhacke QOS_COLOR_GREEN);
344c97bf8c3SAlexander Motin
345c97bf8c3SAlexander Motin /* If not green and mode is SHAPE, enqueue packet. */
346c97bf8c3SAlexander Motin } else if (hinfo->conf.mode == NG_CAR_SHAPE) {
347c97bf8c3SAlexander Motin ng_car_enqueue(hinfo, item);
348c97bf8c3SAlexander Motin return (0);
349c97bf8c3SAlexander Motin
350c97bf8c3SAlexander Motin /* If not green and mode is RED, calculate probability. */
351c97bf8c3SAlexander Motin } else if (hinfo->conf.mode == NG_CAR_RED) {
352c97bf8c3SAlexander Motin /* Is packet is bigger then extended burst? */
353d0d2e523SLutz Donnerhacke if (len - (hinfo->tc - len) > hinfo->conf.ebs ||
354d0d2e523SLutz Donnerhacke col >= QOS_COLOR_RED) {
355ae1be01fSAlexander Motin /* This packet is definitely red. */
356c97bf8c3SAlexander Motin ++hinfo->stats.red_pkts;
357c97bf8c3SAlexander Motin hinfo->te = 0;
358d0d2e523SLutz Donnerhacke NG_CAR_PERFORM_MATCH_ACTION(
359d0d2e523SLutz Donnerhacke hinfo->conf.red_action,
360d0d2e523SLutz Donnerhacke QOS_COLOR_RED);
361c97bf8c3SAlexander Motin
362ae1be01fSAlexander Motin /* Use token bucket to simulate RED-like drop
363ae1be01fSAlexander Motin probability. */
364d0d2e523SLutz Donnerhacke } else if (hinfo->te + (len - hinfo->tc) < hinfo->conf.ebs &&
365d0d2e523SLutz Donnerhacke col <= QOS_COLOR_YELLOW) {
366c97bf8c3SAlexander Motin /* This packet is yellow */
367c97bf8c3SAlexander Motin ++hinfo->stats.yellow_pkts;
368673f5a8bSAlexander Motin hinfo->te += len - hinfo->tc;
369ae1be01fSAlexander Motin /* Go to negative tokens. */
370673f5a8bSAlexander Motin hinfo->tc -= len;
371d0d2e523SLutz Donnerhacke NG_CAR_PERFORM_MATCH_ACTION(
372d0d2e523SLutz Donnerhacke hinfo->conf.yellow_action,
373d0d2e523SLutz Donnerhacke QOS_COLOR_YELLOW);
374c97bf8c3SAlexander Motin } else {
375053359b7SPedro F. Giffuni /* This packet is probably red. */
376c97bf8c3SAlexander Motin ++hinfo->stats.red_pkts;
377c97bf8c3SAlexander Motin hinfo->te = 0;
378d0d2e523SLutz Donnerhacke NG_CAR_PERFORM_MATCH_ACTION(
379d0d2e523SLutz Donnerhacke hinfo->conf.red_action,
380d0d2e523SLutz Donnerhacke QOS_COLOR_RED);
381c97bf8c3SAlexander Motin }
382c97bf8c3SAlexander Motin /* If not green and mode is SINGLE/DOUBLE RATE. */
383c97bf8c3SAlexander Motin } else {
384c97bf8c3SAlexander Motin /* Check extended token bucket. */
385d0d2e523SLutz Donnerhacke if (hinfo->te - len >= 0 && col <= QOS_COLOR_YELLOW) {
386c97bf8c3SAlexander Motin /* This packet is yellow */
387c97bf8c3SAlexander Motin ++hinfo->stats.yellow_pkts;
388673f5a8bSAlexander Motin hinfo->te -= len;
389d0d2e523SLutz Donnerhacke NG_CAR_PERFORM_MATCH_ACTION(
390d0d2e523SLutz Donnerhacke hinfo->conf.yellow_action,
391d0d2e523SLutz Donnerhacke QOS_COLOR_YELLOW);
392c97bf8c3SAlexander Motin } else {
393c97bf8c3SAlexander Motin /* This packet is red */
394c97bf8c3SAlexander Motin ++hinfo->stats.red_pkts;
395d0d2e523SLutz Donnerhacke NG_CAR_PERFORM_MATCH_ACTION(
396d0d2e523SLutz Donnerhacke hinfo->conf.red_action,
397d0d2e523SLutz Donnerhacke QOS_COLOR_RED);
398c97bf8c3SAlexander Motin }
399c97bf8c3SAlexander Motin }
400c97bf8c3SAlexander Motin }
401c97bf8c3SAlexander Motin
402c97bf8c3SAlexander Motin #undef NG_CAR_PERFORM_MATCH_ACTION
403c97bf8c3SAlexander Motin
404c77b232bSAlexander Motin NG_FWD_ITEM_HOOK(error, item, hinfo->dest);
405c97bf8c3SAlexander Motin if (error != 0)
406c97bf8c3SAlexander Motin ++hinfo->stats.errors;
407c97bf8c3SAlexander Motin ++hinfo->stats.passed_pkts;
408c97bf8c3SAlexander Motin
409ae1be01fSAlexander Motin return (error);
410c97bf8c3SAlexander Motin }
411c97bf8c3SAlexander Motin
412c97bf8c3SAlexander Motin /*
413ae1be01fSAlexander Motin * Receive a control message.
414c97bf8c3SAlexander Motin */
415c97bf8c3SAlexander Motin static int
ng_car_rcvmsg(node_p node,item_p item,hook_p lasthook)416c97bf8c3SAlexander Motin ng_car_rcvmsg(node_p node, item_p item, hook_p lasthook)
417c97bf8c3SAlexander Motin {
418c97bf8c3SAlexander Motin const priv_p priv = NG_NODE_PRIVATE(node);
419c97bf8c3SAlexander Motin struct ng_mesg *resp = NULL;
420c97bf8c3SAlexander Motin int error = 0;
421c97bf8c3SAlexander Motin struct ng_mesg *msg;
422c97bf8c3SAlexander Motin
423c97bf8c3SAlexander Motin NGI_GET_MSG(item, msg);
424c97bf8c3SAlexander Motin switch (msg->header.typecookie) {
425c97bf8c3SAlexander Motin case NGM_CAR_COOKIE:
426c97bf8c3SAlexander Motin switch (msg->header.cmd) {
427c97bf8c3SAlexander Motin case NGM_CAR_GET_STATS:
428c97bf8c3SAlexander Motin case NGM_CAR_GETCLR_STATS:
429c97bf8c3SAlexander Motin {
430c97bf8c3SAlexander Motin struct ng_car_bulkstats *bstats;
431c97bf8c3SAlexander Motin
432c97bf8c3SAlexander Motin NG_MKRESPONSE(resp, msg,
433c97bf8c3SAlexander Motin sizeof(*bstats), M_NOWAIT);
434c97bf8c3SAlexander Motin if (resp == NULL) {
435c97bf8c3SAlexander Motin error = ENOMEM;
436c97bf8c3SAlexander Motin break;
437c97bf8c3SAlexander Motin }
438c97bf8c3SAlexander Motin bstats = (struct ng_car_bulkstats *)resp->data;
439c97bf8c3SAlexander Motin
440c97bf8c3SAlexander Motin bcopy(&priv->upper.stats, &bstats->downstream,
441c97bf8c3SAlexander Motin sizeof(bstats->downstream));
442c97bf8c3SAlexander Motin bcopy(&priv->lower.stats, &bstats->upstream,
443c97bf8c3SAlexander Motin sizeof(bstats->upstream));
444c97bf8c3SAlexander Motin }
445c97bf8c3SAlexander Motin if (msg->header.cmd == NGM_CAR_GET_STATS)
446c97bf8c3SAlexander Motin break;
447c97bf8c3SAlexander Motin case NGM_CAR_CLR_STATS:
448c97bf8c3SAlexander Motin bzero(&priv->upper.stats,
449c97bf8c3SAlexander Motin sizeof(priv->upper.stats));
450c97bf8c3SAlexander Motin bzero(&priv->lower.stats,
451c97bf8c3SAlexander Motin sizeof(priv->lower.stats));
452c97bf8c3SAlexander Motin break;
453c97bf8c3SAlexander Motin case NGM_CAR_GET_CONF:
454c97bf8c3SAlexander Motin {
455c97bf8c3SAlexander Motin struct ng_car_bulkconf *bconf;
456c97bf8c3SAlexander Motin
457c97bf8c3SAlexander Motin NG_MKRESPONSE(resp, msg,
458c97bf8c3SAlexander Motin sizeof(*bconf), M_NOWAIT);
459c97bf8c3SAlexander Motin if (resp == NULL) {
460c97bf8c3SAlexander Motin error = ENOMEM;
461c97bf8c3SAlexander Motin break;
462c97bf8c3SAlexander Motin }
463c97bf8c3SAlexander Motin bconf = (struct ng_car_bulkconf *)resp->data;
464c97bf8c3SAlexander Motin
465c97bf8c3SAlexander Motin bcopy(&priv->upper.conf, &bconf->downstream,
466c97bf8c3SAlexander Motin sizeof(bconf->downstream));
467c97bf8c3SAlexander Motin bcopy(&priv->lower.conf, &bconf->upstream,
468c97bf8c3SAlexander Motin sizeof(bconf->upstream));
469673f5a8bSAlexander Motin /* Convert internal 1/(8*128) of pps into pps */
470673f5a8bSAlexander Motin if (bconf->downstream.opt & NG_CAR_COUNT_PACKETS) {
471673f5a8bSAlexander Motin bconf->downstream.cir /= 1024;
472673f5a8bSAlexander Motin bconf->downstream.pir /= 1024;
473673f5a8bSAlexander Motin bconf->downstream.cbs /= 128;
474673f5a8bSAlexander Motin bconf->downstream.ebs /= 128;
475673f5a8bSAlexander Motin }
476673f5a8bSAlexander Motin if (bconf->upstream.opt & NG_CAR_COUNT_PACKETS) {
477673f5a8bSAlexander Motin bconf->upstream.cir /= 1024;
478673f5a8bSAlexander Motin bconf->upstream.pir /= 1024;
479673f5a8bSAlexander Motin bconf->upstream.cbs /= 128;
480673f5a8bSAlexander Motin bconf->upstream.ebs /= 128;
481673f5a8bSAlexander Motin }
482c97bf8c3SAlexander Motin }
483c97bf8c3SAlexander Motin break;
484c97bf8c3SAlexander Motin case NGM_CAR_SET_CONF:
485c97bf8c3SAlexander Motin {
486c97bf8c3SAlexander Motin struct ng_car_bulkconf *const bconf =
487c97bf8c3SAlexander Motin (struct ng_car_bulkconf *)msg->data;
488c97bf8c3SAlexander Motin
489c97bf8c3SAlexander Motin /* Check for invalid or illegal config. */
490673f5a8bSAlexander Motin if (msg->header.arglen != sizeof(*bconf)) {
491673f5a8bSAlexander Motin error = EINVAL;
492673f5a8bSAlexander Motin break;
493673f5a8bSAlexander Motin }
494673f5a8bSAlexander Motin /* Convert pps into internal 1/(8*128) of pps */
495673f5a8bSAlexander Motin if (bconf->downstream.opt & NG_CAR_COUNT_PACKETS) {
496673f5a8bSAlexander Motin bconf->downstream.cir *= 1024;
497673f5a8bSAlexander Motin bconf->downstream.pir *= 1024;
4984a683076SAlexander Motin bconf->downstream.cbs *= 128;
4994a683076SAlexander Motin bconf->downstream.ebs *= 128;
500673f5a8bSAlexander Motin }
501673f5a8bSAlexander Motin if (bconf->upstream.opt & NG_CAR_COUNT_PACKETS) {
502673f5a8bSAlexander Motin bconf->upstream.cir *= 1024;
503673f5a8bSAlexander Motin bconf->upstream.pir *= 1024;
5044a683076SAlexander Motin bconf->upstream.cbs *= 128;
5054a683076SAlexander Motin bconf->upstream.ebs *= 128;
506673f5a8bSAlexander Motin }
507673f5a8bSAlexander Motin if ((bconf->downstream.cir > 1000000000) ||
508673f5a8bSAlexander Motin (bconf->downstream.pir > 1000000000) ||
509673f5a8bSAlexander Motin (bconf->upstream.cir > 1000000000) ||
510673f5a8bSAlexander Motin (bconf->upstream.pir > 1000000000) ||
511673f5a8bSAlexander Motin (bconf->downstream.cbs == 0 &&
512673f5a8bSAlexander Motin bconf->downstream.ebs == 0) ||
513673f5a8bSAlexander Motin (bconf->upstream.cbs == 0 &&
514673f5a8bSAlexander Motin bconf->upstream.ebs == 0))
515c97bf8c3SAlexander Motin {
516c97bf8c3SAlexander Motin error = EINVAL;
517c97bf8c3SAlexander Motin break;
518c97bf8c3SAlexander Motin }
519673f5a8bSAlexander Motin if ((bconf->upstream.mode == NG_CAR_SHAPE) &&
520673f5a8bSAlexander Motin (bconf->upstream.cir == 0)) {
521673f5a8bSAlexander Motin error = EINVAL;
522673f5a8bSAlexander Motin break;
523673f5a8bSAlexander Motin }
524673f5a8bSAlexander Motin if ((bconf->downstream.mode == NG_CAR_SHAPE) &&
525673f5a8bSAlexander Motin (bconf->downstream.cir == 0)) {
526673f5a8bSAlexander Motin error = EINVAL;
527673f5a8bSAlexander Motin break;
528673f5a8bSAlexander Motin }
529c97bf8c3SAlexander Motin
530c97bf8c3SAlexander Motin /* Copy downstream config. */
531c97bf8c3SAlexander Motin bcopy(&bconf->downstream, &priv->upper.conf,
532c97bf8c3SAlexander Motin sizeof(priv->upper.conf));
533c97bf8c3SAlexander Motin priv->upper.tc = priv->upper.conf.cbs;
534c97bf8c3SAlexander Motin if (priv->upper.conf.mode == NG_CAR_RED ||
535b50ace73SAlexander Motin priv->upper.conf.mode == NG_CAR_SHAPE) {
536c97bf8c3SAlexander Motin priv->upper.te = 0;
537c97bf8c3SAlexander Motin } else {
538c97bf8c3SAlexander Motin priv->upper.te = priv->upper.conf.ebs;
539c97bf8c3SAlexander Motin }
540c97bf8c3SAlexander Motin
541c97bf8c3SAlexander Motin /* Copy upstream config. */
542c97bf8c3SAlexander Motin bcopy(&bconf->upstream, &priv->lower.conf,
543c97bf8c3SAlexander Motin sizeof(priv->lower.conf));
544c97bf8c3SAlexander Motin priv->lower.tc = priv->lower.conf.cbs;
545c97bf8c3SAlexander Motin if (priv->lower.conf.mode == NG_CAR_RED ||
546c97bf8c3SAlexander Motin priv->lower.conf.mode == NG_CAR_SHAPE) {
547c97bf8c3SAlexander Motin priv->lower.te = 0;
548c97bf8c3SAlexander Motin } else {
549c97bf8c3SAlexander Motin priv->lower.te = priv->lower.conf.ebs;
550c97bf8c3SAlexander Motin }
551c97bf8c3SAlexander Motin }
552c97bf8c3SAlexander Motin break;
553c97bf8c3SAlexander Motin default:
554c97bf8c3SAlexander Motin error = EINVAL;
555c97bf8c3SAlexander Motin break;
556c97bf8c3SAlexander Motin }
557c97bf8c3SAlexander Motin break;
558c97bf8c3SAlexander Motin default:
559c97bf8c3SAlexander Motin error = EINVAL;
560c97bf8c3SAlexander Motin break;
561c97bf8c3SAlexander Motin }
562c97bf8c3SAlexander Motin NG_RESPOND_MSG(error, node, item, resp);
563c97bf8c3SAlexander Motin NG_FREE_MSG(msg);
564c97bf8c3SAlexander Motin return (error);
565c97bf8c3SAlexander Motin }
566c97bf8c3SAlexander Motin
567c97bf8c3SAlexander Motin /*
568c97bf8c3SAlexander Motin * Do local shutdown processing.
569c97bf8c3SAlexander Motin */
570c97bf8c3SAlexander Motin static int
ng_car_shutdown(node_p node)571c97bf8c3SAlexander Motin ng_car_shutdown(node_p node)
572c97bf8c3SAlexander Motin {
573c97bf8c3SAlexander Motin const priv_p priv = NG_NODE_PRIVATE(node);
574c97bf8c3SAlexander Motin
575df01e689SAlexander Motin ng_uncallout(&priv->upper.q_callout, node);
576df01e689SAlexander Motin ng_uncallout(&priv->lower.q_callout, node);
577c97bf8c3SAlexander Motin mtx_destroy(&priv->upper.q_mtx);
578c97bf8c3SAlexander Motin mtx_destroy(&priv->lower.q_mtx);
579c97bf8c3SAlexander Motin NG_NODE_UNREF(priv->node);
580*9b8db664SDmitry Lukhtionov free(priv, M_NETGRAPH_CAR);
581c97bf8c3SAlexander Motin return (0);
582c97bf8c3SAlexander Motin }
583c97bf8c3SAlexander Motin
584c97bf8c3SAlexander Motin /*
585c97bf8c3SAlexander Motin * Hook disconnection.
586c97bf8c3SAlexander Motin *
587c97bf8c3SAlexander Motin * For this type, removal of the last link destroys the node.
588c97bf8c3SAlexander Motin */
589c97bf8c3SAlexander Motin static int
ng_car_disconnect(hook_p hook)590c97bf8c3SAlexander Motin ng_car_disconnect(hook_p hook)
591c97bf8c3SAlexander Motin {
592c97bf8c3SAlexander Motin struct hookinfo *const hinfo = NG_HOOK_PRIVATE(hook);
593c97bf8c3SAlexander Motin const node_p node = NG_HOOK_NODE(hook);
594c97bf8c3SAlexander Motin const priv_p priv = NG_NODE_PRIVATE(node);
595c97bf8c3SAlexander Motin
596c97bf8c3SAlexander Motin if (hinfo) {
597c97bf8c3SAlexander Motin /* Purge queue if not empty. */
598c97bf8c3SAlexander Motin while (hinfo->q_first != hinfo->q_last) {
599c97bf8c3SAlexander Motin NG_FREE_M(hinfo->q[hinfo->q_first]);
600c97bf8c3SAlexander Motin hinfo->q_first++;
601c97bf8c3SAlexander Motin if (hinfo->q_first >= NG_CAR_QUEUE_SIZE)
602c97bf8c3SAlexander Motin hinfo->q_first = 0;
603c97bf8c3SAlexander Motin }
604c97bf8c3SAlexander Motin /* Remove hook refs. */
605c97bf8c3SAlexander Motin if (hinfo->hook == priv->upper.hook)
606c97bf8c3SAlexander Motin priv->lower.dest = NULL;
607c97bf8c3SAlexander Motin else
608c97bf8c3SAlexander Motin priv->upper.dest = NULL;
609c97bf8c3SAlexander Motin hinfo->hook = NULL;
610c97bf8c3SAlexander Motin }
611c97bf8c3SAlexander Motin /* Already shutting down? */
612c97bf8c3SAlexander Motin if ((NG_NODE_NUMHOOKS(NG_HOOK_NODE(hook)) == 0)
613c97bf8c3SAlexander Motin && (NG_NODE_IS_VALID(NG_HOOK_NODE(hook))))
614c97bf8c3SAlexander Motin ng_rmnode_self(NG_HOOK_NODE(hook));
615c97bf8c3SAlexander Motin return (0);
616c97bf8c3SAlexander Motin }
617c97bf8c3SAlexander Motin
618c97bf8c3SAlexander Motin /*
619c97bf8c3SAlexander Motin * Hook's token buckets refillment.
620c97bf8c3SAlexander Motin */
621c97bf8c3SAlexander Motin static void
ng_car_refillhook(struct hookinfo * h)622c97bf8c3SAlexander Motin ng_car_refillhook(struct hookinfo *h)
623c97bf8c3SAlexander Motin {
624c86d865eSAlexander Motin struct bintime newt, deltat;
625c86d865eSAlexander Motin unsigned int deltat_us;
626c97bf8c3SAlexander Motin
627c97bf8c3SAlexander Motin /* Get current time. */
628c86d865eSAlexander Motin getbinuptime(&newt);
629c86d865eSAlexander Motin
630c86d865eSAlexander Motin /* Get time delta since last refill. */
631c86d865eSAlexander Motin deltat = newt;
632c86d865eSAlexander Motin bintime_sub(&deltat, &h->lastRefill);
633c97bf8c3SAlexander Motin
634c97bf8c3SAlexander Motin /* Time must go forward. */
635c86d865eSAlexander Motin if (deltat.sec < 0) {
636c97bf8c3SAlexander Motin h->lastRefill = newt;
637c97bf8c3SAlexander Motin return;
638c97bf8c3SAlexander Motin }
639c97bf8c3SAlexander Motin
640c86d865eSAlexander Motin /* But not too far forward. */
641c86d865eSAlexander Motin if (deltat.sec >= 1000) {
642c86d865eSAlexander Motin deltat_us = (1000 << 20);
643c97bf8c3SAlexander Motin } else {
644c86d865eSAlexander Motin /* convert bintime to the 1/(2^20) of sec */
645c86d865eSAlexander Motin deltat_us = (deltat.sec << 20) + (deltat.frac >> 44);
646c97bf8c3SAlexander Motin }
647c97bf8c3SAlexander Motin
648c97bf8c3SAlexander Motin if (h->conf.mode == NG_CAR_SINGLE_RATE) {
649c86d865eSAlexander Motin int64_t delta;
650053359b7SPedro F. Giffuni /* Refill committed token bucket. */
651c86d865eSAlexander Motin h->tc += (h->conf.cir * deltat_us) >> 23;
652c97bf8c3SAlexander Motin delta = h->tc - h->conf.cbs;
653c97bf8c3SAlexander Motin if (delta > 0) {
654c97bf8c3SAlexander Motin h->tc = h->conf.cbs;
655c97bf8c3SAlexander Motin
656c97bf8c3SAlexander Motin /* Refill exceeded token bucket. */
657c97bf8c3SAlexander Motin h->te += delta;
658c86d865eSAlexander Motin if (h->te > ((int64_t)h->conf.ebs))
659c97bf8c3SAlexander Motin h->te = h->conf.ebs;
660c97bf8c3SAlexander Motin }
661c97bf8c3SAlexander Motin
662c97bf8c3SAlexander Motin } else if (h->conf.mode == NG_CAR_DOUBLE_RATE) {
663053359b7SPedro F. Giffuni /* Refill committed token bucket. */
664c86d865eSAlexander Motin h->tc += (h->conf.cir * deltat_us) >> 23;
665c86d865eSAlexander Motin if (h->tc > ((int64_t)h->conf.cbs))
666c97bf8c3SAlexander Motin h->tc = h->conf.cbs;
667c97bf8c3SAlexander Motin
668c97bf8c3SAlexander Motin /* Refill peak token bucket. */
669c86d865eSAlexander Motin h->te += (h->conf.pir * deltat_us) >> 23;
670c86d865eSAlexander Motin if (h->te > ((int64_t)h->conf.ebs))
671c97bf8c3SAlexander Motin h->te = h->conf.ebs;
672c97bf8c3SAlexander Motin
673c97bf8c3SAlexander Motin } else { /* RED or SHAPE mode. */
674053359b7SPedro F. Giffuni /* Refill committed token bucket. */
675c86d865eSAlexander Motin h->tc += (h->conf.cir * deltat_us) >> 23;
676c97bf8c3SAlexander Motin if (h->tc > ((int64_t)h->conf.cbs))
677c97bf8c3SAlexander Motin h->tc = h->conf.cbs;
678c97bf8c3SAlexander Motin }
679c97bf8c3SAlexander Motin
680c97bf8c3SAlexander Motin /* Remember this moment. */
681c97bf8c3SAlexander Motin h->lastRefill = newt;
682c97bf8c3SAlexander Motin }
683c97bf8c3SAlexander Motin
684c97bf8c3SAlexander Motin /*
685c97bf8c3SAlexander Motin * Schedule callout when we will have required tokens.
686c97bf8c3SAlexander Motin */
687c97bf8c3SAlexander Motin static void
ng_car_schedule(struct hookinfo * hinfo)688c97bf8c3SAlexander Motin ng_car_schedule(struct hookinfo *hinfo)
689c97bf8c3SAlexander Motin {
690c97bf8c3SAlexander Motin int delay;
691c97bf8c3SAlexander Motin
692c97bf8c3SAlexander Motin delay = (-(hinfo->tc)) * hz * 8 / hinfo->conf.cir + 1;
693c97bf8c3SAlexander Motin
694c97bf8c3SAlexander Motin ng_callout(&hinfo->q_callout, NG_HOOK_NODE(hinfo->hook), hinfo->hook,
695c97bf8c3SAlexander Motin delay, &ng_car_q_event, NULL, 0);
696c97bf8c3SAlexander Motin }
697c97bf8c3SAlexander Motin
698c97bf8c3SAlexander Motin /*
699c97bf8c3SAlexander Motin * Queue processing callout handler.
700c97bf8c3SAlexander Motin */
701c97bf8c3SAlexander Motin void
ng_car_q_event(node_p node,hook_p hook,void * arg,int arg2)702c97bf8c3SAlexander Motin ng_car_q_event(node_p node, hook_p hook, void *arg, int arg2)
703c97bf8c3SAlexander Motin {
704c97bf8c3SAlexander Motin struct hookinfo *hinfo = NG_HOOK_PRIVATE(hook);
705c97bf8c3SAlexander Motin struct mbuf *m;
706c97bf8c3SAlexander Motin int error;
707c97bf8c3SAlexander Motin
708c97bf8c3SAlexander Motin /* Refill tokens for time we have slept. */
709c97bf8c3SAlexander Motin ng_car_refillhook(hinfo);
710c97bf8c3SAlexander Motin
711c97bf8c3SAlexander Motin /* If we have some tokens */
712c97bf8c3SAlexander Motin while (hinfo->tc >= 0) {
713c97bf8c3SAlexander Motin /* Send packet. */
714c97bf8c3SAlexander Motin m = hinfo->q[hinfo->q_first];
715c77b232bSAlexander Motin NG_SEND_DATA_ONLY(error, hinfo->dest, m);
716c77b232bSAlexander Motin if (error != 0)
717c77b232bSAlexander Motin ++hinfo->stats.errors;
718c77b232bSAlexander Motin ++hinfo->stats.passed_pkts;
719c97bf8c3SAlexander Motin
720c97bf8c3SAlexander Motin /* Get next one. */
721c97bf8c3SAlexander Motin hinfo->q_first++;
722c97bf8c3SAlexander Motin if (hinfo->q_first >= NG_CAR_QUEUE_SIZE)
723c97bf8c3SAlexander Motin hinfo->q_first = 0;
724c97bf8c3SAlexander Motin
725c97bf8c3SAlexander Motin /* Stop if none left. */
726c97bf8c3SAlexander Motin if (hinfo->q_first == hinfo->q_last)
727c97bf8c3SAlexander Motin break;
728c97bf8c3SAlexander Motin
729c97bf8c3SAlexander Motin /* If we have more packet, try it. */
730c97bf8c3SAlexander Motin m = hinfo->q[hinfo->q_first];
731673f5a8bSAlexander Motin if (hinfo->conf.opt & NG_CAR_COUNT_PACKETS) {
732673f5a8bSAlexander Motin hinfo->tc -= 128;
733673f5a8bSAlexander Motin } else {
734c97bf8c3SAlexander Motin hinfo->tc -= m->m_pkthdr.len;
735c97bf8c3SAlexander Motin }
736c97bf8c3SAlexander Motin }
737c97bf8c3SAlexander Motin
738c97bf8c3SAlexander Motin /* If something left */
739ae1be01fSAlexander Motin if (hinfo->q_first != hinfo->q_last)
740ae1be01fSAlexander Motin /* Schedule queue processing. */
741c97bf8c3SAlexander Motin ng_car_schedule(hinfo);
742c97bf8c3SAlexander Motin }
743c97bf8c3SAlexander Motin
744c97bf8c3SAlexander Motin /*
745c97bf8c3SAlexander Motin * Enqueue packet.
746c97bf8c3SAlexander Motin */
747c97bf8c3SAlexander Motin static void
ng_car_enqueue(struct hookinfo * hinfo,item_p item)748c97bf8c3SAlexander Motin ng_car_enqueue(struct hookinfo *hinfo, item_p item)
749c97bf8c3SAlexander Motin {
750c97bf8c3SAlexander Motin struct mbuf *m;
751c97bf8c3SAlexander Motin int len;
752d0d2e523SLutz Donnerhacke struct m_qos_color *colp;
753d0d2e523SLutz Donnerhacke enum qos_color col;
754c97bf8c3SAlexander Motin
755c97bf8c3SAlexander Motin NGI_GET_M(item, m);
756c97bf8c3SAlexander Motin NG_FREE_ITEM(item);
757c97bf8c3SAlexander Motin
758d0d2e523SLutz Donnerhacke /* Determine current color of the packet (default green) */
759d0d2e523SLutz Donnerhacke colp = (void *)m_tag_locate(m, M_QOS_COOKIE, M_QOS_COLOR, NULL);
760d0d2e523SLutz Donnerhacke if ((hinfo->conf.opt & NG_CAR_COLOR_AWARE) && (colp != NULL))
761d0d2e523SLutz Donnerhacke col = colp->color;
762d0d2e523SLutz Donnerhacke else
763d0d2e523SLutz Donnerhacke col = QOS_COLOR_GREEN;
764d0d2e523SLutz Donnerhacke
765ae1be01fSAlexander Motin /* Lock queue mutex. */
766c97bf8c3SAlexander Motin mtx_lock(&hinfo->q_mtx);
767c97bf8c3SAlexander Motin
768ae1be01fSAlexander Motin /* Calculate used queue length. */
769c97bf8c3SAlexander Motin len = hinfo->q_last - hinfo->q_first;
770c97bf8c3SAlexander Motin if (len < 0)
771c97bf8c3SAlexander Motin len += NG_CAR_QUEUE_SIZE;
772c97bf8c3SAlexander Motin
773ae1be01fSAlexander Motin /* If queue is overflowed or we have no RED tokens. */
774c97bf8c3SAlexander Motin if ((len >= (NG_CAR_QUEUE_SIZE - 1)) ||
775d0d2e523SLutz Donnerhacke (hinfo->te + len >= NG_CAR_QUEUE_SIZE) ||
776d0d2e523SLutz Donnerhacke (col >= QOS_COLOR_RED)) {
777ae1be01fSAlexander Motin /* Drop packet. */
778c97bf8c3SAlexander Motin ++hinfo->stats.red_pkts;
779def4e701SGleb Smirnoff ++hinfo->stats.dropped_pkts;
780c97bf8c3SAlexander Motin NG_FREE_M(m);
781c97bf8c3SAlexander Motin
782c97bf8c3SAlexander Motin hinfo->te = 0;
783c97bf8c3SAlexander Motin } else {
784c97bf8c3SAlexander Motin /* This packet is yellow. */
785c97bf8c3SAlexander Motin ++hinfo->stats.yellow_pkts;
786c97bf8c3SAlexander Motin
787c97bf8c3SAlexander Motin /* Enqueue packet. */
788c97bf8c3SAlexander Motin hinfo->q[hinfo->q_last] = m;
789c97bf8c3SAlexander Motin hinfo->q_last++;
790c97bf8c3SAlexander Motin if (hinfo->q_last >= NG_CAR_QUEUE_SIZE)
791c97bf8c3SAlexander Motin hinfo->q_last = 0;
792c97bf8c3SAlexander Motin
793c97bf8c3SAlexander Motin /* Use RED tokens. */
794c97bf8c3SAlexander Motin if (len > NG_CAR_QUEUE_MIN_TH)
795c97bf8c3SAlexander Motin hinfo->te += len - NG_CAR_QUEUE_MIN_TH;
796c97bf8c3SAlexander Motin
797ae1be01fSAlexander Motin /* If this is a first packet in the queue. */
798c97bf8c3SAlexander Motin if (len == 0) {
799673f5a8bSAlexander Motin if (hinfo->conf.opt & NG_CAR_COUNT_PACKETS) {
800673f5a8bSAlexander Motin hinfo->tc -= 128;
801673f5a8bSAlexander Motin } else {
802c97bf8c3SAlexander Motin hinfo->tc -= m->m_pkthdr.len;
803673f5a8bSAlexander Motin }
804c97bf8c3SAlexander Motin
805ae1be01fSAlexander Motin /* Schedule queue processing. */
806c97bf8c3SAlexander Motin ng_car_schedule(hinfo);
807c97bf8c3SAlexander Motin }
808c97bf8c3SAlexander Motin }
809c97bf8c3SAlexander Motin
810ae1be01fSAlexander Motin /* Unlock queue mutex. */
811c97bf8c3SAlexander Motin mtx_unlock(&hinfo->q_mtx);
812c97bf8c3SAlexander Motin }
813