xref: /aosp_15_r20/external/iptables/extensions/libxt_rateest.c (revision a71a954618bbadd4a345637e5edcf36eec826889)
1*a71a9546SAutomerger Merge Worker /*
2*a71a9546SAutomerger Merge Worker  * Copyright (c) 2008-2013 Patrick McHardy <[email protected]>
3*a71a9546SAutomerger Merge Worker  */
4*a71a9546SAutomerger Merge Worker 
5*a71a9546SAutomerger Merge Worker #include <stdbool.h>
6*a71a9546SAutomerger Merge Worker #include <stdio.h>
7*a71a9546SAutomerger Merge Worker #include <string.h>
8*a71a9546SAutomerger Merge Worker #include <stdlib.h>
9*a71a9546SAutomerger Merge Worker #include <stddef.h>
10*a71a9546SAutomerger Merge Worker #include <getopt.h>
11*a71a9546SAutomerger Merge Worker 
12*a71a9546SAutomerger Merge Worker #include <xtables.h>
13*a71a9546SAutomerger Merge Worker #include <linux/netfilter/xt_rateest.h>
14*a71a9546SAutomerger Merge Worker 
rateest_help(void)15*a71a9546SAutomerger Merge Worker static void rateest_help(void)
16*a71a9546SAutomerger Merge Worker {
17*a71a9546SAutomerger Merge Worker 	printf(
18*a71a9546SAutomerger Merge Worker "rateest match options:\n"
19*a71a9546SAutomerger Merge Worker " --rateest1 name		Rate estimator name\n"
20*a71a9546SAutomerger Merge Worker " --rateest2 name		Rate estimator name\n"
21*a71a9546SAutomerger Merge Worker " --rateest-delta		Compare difference(s) to given rate(s)\n"
22*a71a9546SAutomerger Merge Worker " --rateest-bps1 [bps]		Compare bps\n"
23*a71a9546SAutomerger Merge Worker " --rateest-pps1 [pps]		Compare pps\n"
24*a71a9546SAutomerger Merge Worker " --rateest-bps2 [bps]		Compare bps\n"
25*a71a9546SAutomerger Merge Worker " --rateest-pps2 [pps]		Compare pps\n"
26*a71a9546SAutomerger Merge Worker " [!] --rateest-lt		Match if rate is less than given rate/estimator\n"
27*a71a9546SAutomerger Merge Worker " [!] --rateest-gt		Match if rate is greater than given rate/estimator\n"
28*a71a9546SAutomerger Merge Worker " [!] --rateest-eq		Match if rate is equal to given rate/estimator\n");
29*a71a9546SAutomerger Merge Worker }
30*a71a9546SAutomerger Merge Worker 
31*a71a9546SAutomerger Merge Worker enum rateest_options {
32*a71a9546SAutomerger Merge Worker 	OPT_RATEEST1,
33*a71a9546SAutomerger Merge Worker 	OPT_RATEEST2,
34*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_BPS1,
35*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_PPS1,
36*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_BPS2,
37*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_PPS2,
38*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_DELTA,
39*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_LT,
40*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_GT,
41*a71a9546SAutomerger Merge Worker 	OPT_RATEEST_EQ,
42*a71a9546SAutomerger Merge Worker };
43*a71a9546SAutomerger Merge Worker 
44*a71a9546SAutomerger Merge Worker static const struct option rateest_opts[] = {
45*a71a9546SAutomerger Merge Worker 	{.name = "rateest1",      .has_arg = true,  .val = OPT_RATEEST1},
46*a71a9546SAutomerger Merge Worker 	{.name = "rateest",       .has_arg = true,  .val = OPT_RATEEST1}, /* alias for absolute mode */
47*a71a9546SAutomerger Merge Worker 	{.name = "rateest2",      .has_arg = true,  .val = OPT_RATEEST2},
48*a71a9546SAutomerger Merge Worker 	{.name = "rateest-bps1",  .has_arg = false, .val = OPT_RATEEST_BPS1},
49*a71a9546SAutomerger Merge Worker 	{.name = "rateest-pps1",  .has_arg = false, .val = OPT_RATEEST_PPS1},
50*a71a9546SAutomerger Merge Worker 	{.name = "rateest-bps2",  .has_arg = false, .val = OPT_RATEEST_BPS2},
51*a71a9546SAutomerger Merge Worker 	{.name = "rateest-pps2",  .has_arg = false, .val = OPT_RATEEST_PPS2},
52*a71a9546SAutomerger Merge Worker 	{.name = "rateest-bps",   .has_arg = false, .val = OPT_RATEEST_BPS2}, /* alias for absolute mode */
53*a71a9546SAutomerger Merge Worker 	{.name = "rateest-pps",   .has_arg = false, .val = OPT_RATEEST_PPS2}, /* alias for absolute mode */
54*a71a9546SAutomerger Merge Worker 	{.name = "rateest-delta", .has_arg = false, .val = OPT_RATEEST_DELTA},
55*a71a9546SAutomerger Merge Worker 	{.name = "rateest-lt",    .has_arg = false, .val = OPT_RATEEST_LT},
56*a71a9546SAutomerger Merge Worker 	{.name = "rateest-gt",    .has_arg = false, .val = OPT_RATEEST_GT},
57*a71a9546SAutomerger Merge Worker 	{.name = "rateest-eq",    .has_arg = false, .val = OPT_RATEEST_EQ},
58*a71a9546SAutomerger Merge Worker 	XT_GETOPT_TABLEEND,
59*a71a9546SAutomerger Merge Worker };
60*a71a9546SAutomerger Merge Worker 
61*a71a9546SAutomerger Merge Worker /* Copied from iproute. See http://physics.nist.gov/cuu/Units/binary.html */
62*a71a9546SAutomerger Merge Worker static const struct rate_suffix {
63*a71a9546SAutomerger Merge Worker 	const char *name;
64*a71a9546SAutomerger Merge Worker 	double scale;
65*a71a9546SAutomerger Merge Worker } suffixes[] = {
66*a71a9546SAutomerger Merge Worker 	{ "bit",	1. },
67*a71a9546SAutomerger Merge Worker 	{ "Kibit",	1024. },
68*a71a9546SAutomerger Merge Worker 	{ "kbit",	1000. },
69*a71a9546SAutomerger Merge Worker 	{ "Mibit",	1024.*1024. },
70*a71a9546SAutomerger Merge Worker 	{ "mbit",	1000000. },
71*a71a9546SAutomerger Merge Worker 	{ "Gibit",	1024.*1024.*1024. },
72*a71a9546SAutomerger Merge Worker 	{ "gbit",	1000000000. },
73*a71a9546SAutomerger Merge Worker 	{ "Tibit",	1024.*1024.*1024.*1024. },
74*a71a9546SAutomerger Merge Worker 	{ "tbit",	1000000000000. },
75*a71a9546SAutomerger Merge Worker 	{ "Bps",	8. },
76*a71a9546SAutomerger Merge Worker 	{ "KiBps",	8.*1024. },
77*a71a9546SAutomerger Merge Worker 	{ "KBps",	8000. },
78*a71a9546SAutomerger Merge Worker 	{ "MiBps",	8.*1024*1024. },
79*a71a9546SAutomerger Merge Worker 	{ "MBps",	8000000. },
80*a71a9546SAutomerger Merge Worker 	{ "GiBps",	8.*1024.*1024.*1024. },
81*a71a9546SAutomerger Merge Worker 	{ "GBps",	8000000000. },
82*a71a9546SAutomerger Merge Worker 	{ "TiBps",	8.*1024.*1024.*1024.*1024. },
83*a71a9546SAutomerger Merge Worker 	{ "TBps",	8000000000000. },
84*a71a9546SAutomerger Merge Worker 	{NULL},
85*a71a9546SAutomerger Merge Worker };
86*a71a9546SAutomerger Merge Worker 
87*a71a9546SAutomerger Merge Worker static int
rateest_get_rate(uint32_t * rate,const char * str)88*a71a9546SAutomerger Merge Worker rateest_get_rate(uint32_t *rate, const char *str)
89*a71a9546SAutomerger Merge Worker {
90*a71a9546SAutomerger Merge Worker 	char *p;
91*a71a9546SAutomerger Merge Worker 	double bps = strtod(str, &p);
92*a71a9546SAutomerger Merge Worker 	const struct rate_suffix *s;
93*a71a9546SAutomerger Merge Worker 
94*a71a9546SAutomerger Merge Worker 	if (p == str)
95*a71a9546SAutomerger Merge Worker 		return -1;
96*a71a9546SAutomerger Merge Worker 
97*a71a9546SAutomerger Merge Worker 	if (*p == '\0') {
98*a71a9546SAutomerger Merge Worker 		*rate = bps / 8.;	/* assume bytes/sec */
99*a71a9546SAutomerger Merge Worker 		return 0;
100*a71a9546SAutomerger Merge Worker 	}
101*a71a9546SAutomerger Merge Worker 
102*a71a9546SAutomerger Merge Worker 	for (s = suffixes; s->name; ++s) {
103*a71a9546SAutomerger Merge Worker 		if (strcasecmp(s->name, p) == 0) {
104*a71a9546SAutomerger Merge Worker 			*rate = (bps * s->scale) / 8.;
105*a71a9546SAutomerger Merge Worker 			return 0;
106*a71a9546SAutomerger Merge Worker 		}
107*a71a9546SAutomerger Merge Worker 	}
108*a71a9546SAutomerger Merge Worker 
109*a71a9546SAutomerger Merge Worker 	return -1;
110*a71a9546SAutomerger Merge Worker }
111*a71a9546SAutomerger Merge Worker 
112*a71a9546SAutomerger Merge Worker static int
rateest_parse(int c,char ** argv,int invert,unsigned int * flags,const void * entry,struct xt_entry_match ** match)113*a71a9546SAutomerger Merge Worker rateest_parse(int c, char **argv, int invert, unsigned int *flags,
114*a71a9546SAutomerger Merge Worker 	      const void *entry, struct xt_entry_match **match)
115*a71a9546SAutomerger Merge Worker {
116*a71a9546SAutomerger Merge Worker 	struct xt_rateest_match_info *info = (void *)(*match)->data;
117*a71a9546SAutomerger Merge Worker 	unsigned int val;
118*a71a9546SAutomerger Merge Worker 
119*a71a9546SAutomerger Merge Worker 	switch (c) {
120*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST1:
121*a71a9546SAutomerger Merge Worker 		if (invert)
122*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
123*a71a9546SAutomerger Merge Worker 				   "rateest: rateest can't be inverted");
124*a71a9546SAutomerger Merge Worker 
125*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
126*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
127*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify --rateest1 twice");
128*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
129*a71a9546SAutomerger Merge Worker 
130*a71a9546SAutomerger Merge Worker 		strncpy(info->name1, optarg, sizeof(info->name1) - 1);
131*a71a9546SAutomerger Merge Worker 		break;
132*a71a9546SAutomerger Merge Worker 
133*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST2:
134*a71a9546SAutomerger Merge Worker 		if (invert)
135*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
136*a71a9546SAutomerger Merge Worker 				   "rateest: rateest can't be inverted");
137*a71a9546SAutomerger Merge Worker 
138*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
139*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
140*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify --rateest2 twice");
141*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
142*a71a9546SAutomerger Merge Worker 
143*a71a9546SAutomerger Merge Worker 		strncpy(info->name2, optarg, sizeof(info->name2) - 1);
144*a71a9546SAutomerger Merge Worker 		info->flags |= XT_RATEEST_MATCH_REL;
145*a71a9546SAutomerger Merge Worker 		break;
146*a71a9546SAutomerger Merge Worker 
147*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_BPS1:
148*a71a9546SAutomerger Merge Worker 		if (invert)
149*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
150*a71a9546SAutomerger Merge Worker 				   "rateest: rateest-bps can't be inverted");
151*a71a9546SAutomerger Merge Worker 
152*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
153*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
154*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify --rateest-bps1 twice");
155*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
156*a71a9546SAutomerger Merge Worker 
157*a71a9546SAutomerger Merge Worker 		info->flags |= XT_RATEEST_MATCH_BPS;
158*a71a9546SAutomerger Merge Worker 
159*a71a9546SAutomerger Merge Worker 		/* The rate is optional and only required in absolute mode */
160*a71a9546SAutomerger Merge Worker 		if (!argv[optind] || *argv[optind] == '-' || *argv[optind] == '!')
161*a71a9546SAutomerger Merge Worker 			break;
162*a71a9546SAutomerger Merge Worker 
163*a71a9546SAutomerger Merge Worker 		if (rateest_get_rate(&info->bps1, argv[optind]) < 0)
164*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
165*a71a9546SAutomerger Merge Worker 				   "rateest: could not parse rate `%s'",
166*a71a9546SAutomerger Merge Worker 				   argv[optind]);
167*a71a9546SAutomerger Merge Worker 		optind++;
168*a71a9546SAutomerger Merge Worker 		break;
169*a71a9546SAutomerger Merge Worker 
170*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_PPS1:
171*a71a9546SAutomerger Merge Worker 		if (invert)
172*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
173*a71a9546SAutomerger Merge Worker 				   "rateest: rateest-pps can't be inverted");
174*a71a9546SAutomerger Merge Worker 
175*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
176*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
177*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify --rateest-pps1 twice");
178*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
179*a71a9546SAutomerger Merge Worker 
180*a71a9546SAutomerger Merge Worker 		info->flags |= XT_RATEEST_MATCH_PPS;
181*a71a9546SAutomerger Merge Worker 
182*a71a9546SAutomerger Merge Worker 		/* The rate is optional and only required in absolute mode */
183*a71a9546SAutomerger Merge Worker 		if (!argv[optind] || *argv[optind] == '-' || *argv[optind] == '!')
184*a71a9546SAutomerger Merge Worker 			break;
185*a71a9546SAutomerger Merge Worker 
186*a71a9546SAutomerger Merge Worker 		if (!xtables_strtoui(argv[optind], NULL, &val, 0, UINT32_MAX))
187*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
188*a71a9546SAutomerger Merge Worker 				   "rateest: could not parse pps `%s'",
189*a71a9546SAutomerger Merge Worker 				   argv[optind]);
190*a71a9546SAutomerger Merge Worker 		info->pps1 = val;
191*a71a9546SAutomerger Merge Worker 		optind++;
192*a71a9546SAutomerger Merge Worker 		break;
193*a71a9546SAutomerger Merge Worker 
194*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_BPS2:
195*a71a9546SAutomerger Merge Worker 		if (invert)
196*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
197*a71a9546SAutomerger Merge Worker 				   "rateest: rateest-bps can't be inverted");
198*a71a9546SAutomerger Merge Worker 
199*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
200*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
201*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify --rateest-bps2 twice");
202*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
203*a71a9546SAutomerger Merge Worker 
204*a71a9546SAutomerger Merge Worker 		info->flags |= XT_RATEEST_MATCH_BPS;
205*a71a9546SAutomerger Merge Worker 
206*a71a9546SAutomerger Merge Worker 		/* The rate is optional and only required in absolute mode */
207*a71a9546SAutomerger Merge Worker 		if (!argv[optind] || *argv[optind] == '-' || *argv[optind] == '!')
208*a71a9546SAutomerger Merge Worker 			break;
209*a71a9546SAutomerger Merge Worker 
210*a71a9546SAutomerger Merge Worker 		if (rateest_get_rate(&info->bps2, argv[optind]) < 0)
211*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
212*a71a9546SAutomerger Merge Worker 				   "rateest: could not parse rate `%s'",
213*a71a9546SAutomerger Merge Worker 				   argv[optind]);
214*a71a9546SAutomerger Merge Worker 		optind++;
215*a71a9546SAutomerger Merge Worker 		break;
216*a71a9546SAutomerger Merge Worker 
217*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_PPS2:
218*a71a9546SAutomerger Merge Worker 		if (invert)
219*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
220*a71a9546SAutomerger Merge Worker 				   "rateest: rateest-pps can't be inverted");
221*a71a9546SAutomerger Merge Worker 
222*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
223*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
224*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify --rateest-pps2 twice");
225*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
226*a71a9546SAutomerger Merge Worker 
227*a71a9546SAutomerger Merge Worker 		info->flags |= XT_RATEEST_MATCH_PPS;
228*a71a9546SAutomerger Merge Worker 
229*a71a9546SAutomerger Merge Worker 		/* The rate is optional and only required in absolute mode */
230*a71a9546SAutomerger Merge Worker 		if (!argv[optind] || *argv[optind] == '-' || *argv[optind] == '!')
231*a71a9546SAutomerger Merge Worker 			break;
232*a71a9546SAutomerger Merge Worker 
233*a71a9546SAutomerger Merge Worker 		if (!xtables_strtoui(argv[optind], NULL, &val, 0, UINT32_MAX))
234*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
235*a71a9546SAutomerger Merge Worker 				   "rateest: could not parse pps `%s'",
236*a71a9546SAutomerger Merge Worker 				   argv[optind]);
237*a71a9546SAutomerger Merge Worker 		info->pps2 = val;
238*a71a9546SAutomerger Merge Worker 		optind++;
239*a71a9546SAutomerger Merge Worker 		break;
240*a71a9546SAutomerger Merge Worker 
241*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_DELTA:
242*a71a9546SAutomerger Merge Worker 		if (invert)
243*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
244*a71a9546SAutomerger Merge Worker 				   "rateest: rateest-delta can't be inverted");
245*a71a9546SAutomerger Merge Worker 
246*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
247*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
248*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify --rateest-delta twice");
249*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
250*a71a9546SAutomerger Merge Worker 
251*a71a9546SAutomerger Merge Worker 		info->flags |= XT_RATEEST_MATCH_DELTA;
252*a71a9546SAutomerger Merge Worker 		break;
253*a71a9546SAutomerger Merge Worker 
254*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_EQ:
255*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
256*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
257*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify lt/gt/eq twice");
258*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
259*a71a9546SAutomerger Merge Worker 
260*a71a9546SAutomerger Merge Worker 		info->mode = XT_RATEEST_MATCH_EQ;
261*a71a9546SAutomerger Merge Worker 		if (invert)
262*a71a9546SAutomerger Merge Worker 			info->flags |= XT_RATEEST_MATCH_INVERT;
263*a71a9546SAutomerger Merge Worker 		break;
264*a71a9546SAutomerger Merge Worker 
265*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_LT:
266*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
267*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
268*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify lt/gt/eq twice");
269*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
270*a71a9546SAutomerger Merge Worker 
271*a71a9546SAutomerger Merge Worker 		info->mode = XT_RATEEST_MATCH_LT;
272*a71a9546SAutomerger Merge Worker 		if (invert)
273*a71a9546SAutomerger Merge Worker 			info->flags |= XT_RATEEST_MATCH_INVERT;
274*a71a9546SAutomerger Merge Worker 		break;
275*a71a9546SAutomerger Merge Worker 
276*a71a9546SAutomerger Merge Worker 	case OPT_RATEEST_GT:
277*a71a9546SAutomerger Merge Worker 		if (*flags & (1 << c))
278*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
279*a71a9546SAutomerger Merge Worker 				   "rateest: can't specify lt/gt/eq twice");
280*a71a9546SAutomerger Merge Worker 		*flags |= 1 << c;
281*a71a9546SAutomerger Merge Worker 
282*a71a9546SAutomerger Merge Worker 		info->mode = XT_RATEEST_MATCH_GT;
283*a71a9546SAutomerger Merge Worker 		if (invert)
284*a71a9546SAutomerger Merge Worker 			info->flags |= XT_RATEEST_MATCH_INVERT;
285*a71a9546SAutomerger Merge Worker 		break;
286*a71a9546SAutomerger Merge Worker 	}
287*a71a9546SAutomerger Merge Worker 
288*a71a9546SAutomerger Merge Worker 	return 1;
289*a71a9546SAutomerger Merge Worker }
290*a71a9546SAutomerger Merge Worker 
rateest_final_check(struct xt_fcheck_call * cb)291*a71a9546SAutomerger Merge Worker static void rateest_final_check(struct xt_fcheck_call *cb)
292*a71a9546SAutomerger Merge Worker {
293*a71a9546SAutomerger Merge Worker 	struct xt_rateest_match_info *info = cb->data;
294*a71a9546SAutomerger Merge Worker 
295*a71a9546SAutomerger Merge Worker 	if (info == NULL)
296*a71a9546SAutomerger Merge Worker 		xtables_error(PARAMETER_PROBLEM, "rateest match: "
297*a71a9546SAutomerger Merge Worker 		           "you need to specify some flags");
298*a71a9546SAutomerger Merge Worker 	if (!(info->flags & XT_RATEEST_MATCH_REL))
299*a71a9546SAutomerger Merge Worker 		info->flags |= XT_RATEEST_MATCH_ABS;
300*a71a9546SAutomerger Merge Worker }
301*a71a9546SAutomerger Merge Worker 
302*a71a9546SAutomerger Merge Worker static void
rateest_print_rate(uint32_t rate,int numeric)303*a71a9546SAutomerger Merge Worker rateest_print_rate(uint32_t rate, int numeric)
304*a71a9546SAutomerger Merge Worker {
305*a71a9546SAutomerger Merge Worker 	double tmp = (double)rate*8;
306*a71a9546SAutomerger Merge Worker 
307*a71a9546SAutomerger Merge Worker 	if (numeric)
308*a71a9546SAutomerger Merge Worker 		printf(" %u", rate);
309*a71a9546SAutomerger Merge Worker 	else if (tmp >= 1000.0*1000000.0)
310*a71a9546SAutomerger Merge Worker 		printf(" %.0fMbit", tmp/1000000.0);
311*a71a9546SAutomerger Merge Worker 	else if (tmp >= 1000.0 * 1000.0)
312*a71a9546SAutomerger Merge Worker 		printf(" %.0fKbit", tmp/1000.0);
313*a71a9546SAutomerger Merge Worker 	else
314*a71a9546SAutomerger Merge Worker 		printf(" %.0fbit", tmp);
315*a71a9546SAutomerger Merge Worker }
316*a71a9546SAutomerger Merge Worker 
317*a71a9546SAutomerger Merge Worker static void
rateest_print_mode(const struct xt_rateest_match_info * info,const char * prefix)318*a71a9546SAutomerger Merge Worker rateest_print_mode(const struct xt_rateest_match_info *info,
319*a71a9546SAutomerger Merge Worker                    const char *prefix)
320*a71a9546SAutomerger Merge Worker {
321*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_INVERT)
322*a71a9546SAutomerger Merge Worker 		printf(" !");
323*a71a9546SAutomerger Merge Worker 
324*a71a9546SAutomerger Merge Worker 	switch (info->mode) {
325*a71a9546SAutomerger Merge Worker 	case XT_RATEEST_MATCH_EQ:
326*a71a9546SAutomerger Merge Worker 		printf(" %seq", prefix);
327*a71a9546SAutomerger Merge Worker 		break;
328*a71a9546SAutomerger Merge Worker 	case XT_RATEEST_MATCH_LT:
329*a71a9546SAutomerger Merge Worker 		printf(" %slt", prefix);
330*a71a9546SAutomerger Merge Worker 		break;
331*a71a9546SAutomerger Merge Worker 	case XT_RATEEST_MATCH_GT:
332*a71a9546SAutomerger Merge Worker 		printf(" %sgt", prefix);
333*a71a9546SAutomerger Merge Worker 		break;
334*a71a9546SAutomerger Merge Worker 	default:
335*a71a9546SAutomerger Merge Worker 		exit(1);
336*a71a9546SAutomerger Merge Worker 	}
337*a71a9546SAutomerger Merge Worker }
338*a71a9546SAutomerger Merge Worker 
339*a71a9546SAutomerger Merge Worker static void
rateest_print(const void * ip,const struct xt_entry_match * match,int numeric)340*a71a9546SAutomerger Merge Worker rateest_print(const void *ip, const struct xt_entry_match *match, int numeric)
341*a71a9546SAutomerger Merge Worker {
342*a71a9546SAutomerger Merge Worker 	const struct xt_rateest_match_info *info = (const void *)match->data;
343*a71a9546SAutomerger Merge Worker 
344*a71a9546SAutomerger Merge Worker 	printf(" rateest match ");
345*a71a9546SAutomerger Merge Worker 
346*a71a9546SAutomerger Merge Worker 	printf("%s", info->name1);
347*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_DELTA)
348*a71a9546SAutomerger Merge Worker 		printf(" delta");
349*a71a9546SAutomerger Merge Worker 
350*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_BPS) {
351*a71a9546SAutomerger Merge Worker 		printf(" bps");
352*a71a9546SAutomerger Merge Worker 		if (info->flags & XT_RATEEST_MATCH_DELTA)
353*a71a9546SAutomerger Merge Worker 			rateest_print_rate(info->bps1, numeric);
354*a71a9546SAutomerger Merge Worker 		if (info->flags & XT_RATEEST_MATCH_ABS) {
355*a71a9546SAutomerger Merge Worker 			rateest_print_rate(info->bps2, numeric);
356*a71a9546SAutomerger Merge Worker 			rateest_print_mode(info, "");
357*a71a9546SAutomerger Merge Worker 		}
358*a71a9546SAutomerger Merge Worker 	}
359*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_PPS) {
360*a71a9546SAutomerger Merge Worker 		printf(" pps");
361*a71a9546SAutomerger Merge Worker 		if (info->flags & XT_RATEEST_MATCH_DELTA)
362*a71a9546SAutomerger Merge Worker 			printf(" %u", info->pps1);
363*a71a9546SAutomerger Merge Worker 		if (info->flags & XT_RATEEST_MATCH_ABS) {
364*a71a9546SAutomerger Merge Worker 			rateest_print_mode(info, "");
365*a71a9546SAutomerger Merge Worker 			printf(" %u", info->pps2);
366*a71a9546SAutomerger Merge Worker 		}
367*a71a9546SAutomerger Merge Worker 	}
368*a71a9546SAutomerger Merge Worker 
369*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_REL) {
370*a71a9546SAutomerger Merge Worker 		rateest_print_mode(info, "");
371*a71a9546SAutomerger Merge Worker 
372*a71a9546SAutomerger Merge Worker 		printf(" %s", info->name2);
373*a71a9546SAutomerger Merge Worker 
374*a71a9546SAutomerger Merge Worker 		if (info->flags & XT_RATEEST_MATCH_BPS) {
375*a71a9546SAutomerger Merge Worker 			printf(" bps");
376*a71a9546SAutomerger Merge Worker 			if (info->flags & XT_RATEEST_MATCH_DELTA)
377*a71a9546SAutomerger Merge Worker 				rateest_print_rate(info->bps2, numeric);
378*a71a9546SAutomerger Merge Worker 		}
379*a71a9546SAutomerger Merge Worker 		if (info->flags & XT_RATEEST_MATCH_PPS) {
380*a71a9546SAutomerger Merge Worker 			printf(" pps");
381*a71a9546SAutomerger Merge Worker 			if (info->flags & XT_RATEEST_MATCH_DELTA)
382*a71a9546SAutomerger Merge Worker 				printf(" %u", info->pps2);
383*a71a9546SAutomerger Merge Worker 		}
384*a71a9546SAutomerger Merge Worker 	}
385*a71a9546SAutomerger Merge Worker }
386*a71a9546SAutomerger Merge Worker 
__rateest_save_rate(const struct xt_rateest_match_info * info,const char * name,uint32_t r1,uint32_t r2,int numeric)387*a71a9546SAutomerger Merge Worker static void __rateest_save_rate(const struct xt_rateest_match_info *info,
388*a71a9546SAutomerger Merge Worker                                 const char *name, uint32_t r1, uint32_t r2,
389*a71a9546SAutomerger Merge Worker                                 int numeric)
390*a71a9546SAutomerger Merge Worker {
391*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_DELTA) {
392*a71a9546SAutomerger Merge Worker 		printf(" --rateest-%s1", name);
393*a71a9546SAutomerger Merge Worker 		rateest_print_rate(r1, numeric);
394*a71a9546SAutomerger Merge Worker 		rateest_print_mode(info, "--rateest-");
395*a71a9546SAutomerger Merge Worker 		printf(" --rateest-%s2", name);
396*a71a9546SAutomerger Merge Worker 	} else {
397*a71a9546SAutomerger Merge Worker 		rateest_print_mode(info, "--rateest-");
398*a71a9546SAutomerger Merge Worker 		printf(" --rateest-%s", name);
399*a71a9546SAutomerger Merge Worker 	}
400*a71a9546SAutomerger Merge Worker 
401*a71a9546SAutomerger Merge Worker 	if (info->flags & (XT_RATEEST_MATCH_ABS|XT_RATEEST_MATCH_DELTA))
402*a71a9546SAutomerger Merge Worker 		rateest_print_rate(r2, numeric);
403*a71a9546SAutomerger Merge Worker }
404*a71a9546SAutomerger Merge Worker 
rateest_save_rates(const struct xt_rateest_match_info * info)405*a71a9546SAutomerger Merge Worker static void rateest_save_rates(const struct xt_rateest_match_info *info)
406*a71a9546SAutomerger Merge Worker {
407*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_BPS)
408*a71a9546SAutomerger Merge Worker 		__rateest_save_rate(info, "bps", info->bps1, info->bps2, 0);
409*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_PPS)
410*a71a9546SAutomerger Merge Worker 		__rateest_save_rate(info, "pps", info->pps1, info->pps2, 1);
411*a71a9546SAutomerger Merge Worker }
412*a71a9546SAutomerger Merge Worker 
413*a71a9546SAutomerger Merge Worker 
414*a71a9546SAutomerger Merge Worker static void
rateest_save(const void * ip,const struct xt_entry_match * match)415*a71a9546SAutomerger Merge Worker rateest_save(const void *ip, const struct xt_entry_match *match)
416*a71a9546SAutomerger Merge Worker {
417*a71a9546SAutomerger Merge Worker 	const struct xt_rateest_match_info *info = (const void *)match->data;
418*a71a9546SAutomerger Merge Worker 
419*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_DELTA)
420*a71a9546SAutomerger Merge Worker 		printf(" --rateest-delta");
421*a71a9546SAutomerger Merge Worker 
422*a71a9546SAutomerger Merge Worker 	if (info->flags & XT_RATEEST_MATCH_REL) {
423*a71a9546SAutomerger Merge Worker 		printf(" --rateest1 %s", info->name1);
424*a71a9546SAutomerger Merge Worker 		rateest_save_rates(info);
425*a71a9546SAutomerger Merge Worker 		printf(" --rateest2 %s", info->name2);
426*a71a9546SAutomerger Merge Worker 	} else { /* XT_RATEEST_MATCH_ABS */
427*a71a9546SAutomerger Merge Worker 		printf(" --rateest %s", info->name1);
428*a71a9546SAutomerger Merge Worker 		rateest_save_rates(info);
429*a71a9546SAutomerger Merge Worker 	}
430*a71a9546SAutomerger Merge Worker }
431*a71a9546SAutomerger Merge Worker 
432*a71a9546SAutomerger Merge Worker static struct xtables_match rateest_mt_reg = {
433*a71a9546SAutomerger Merge Worker 	.family		= NFPROTO_UNSPEC,
434*a71a9546SAutomerger Merge Worker 	.name		= "rateest",
435*a71a9546SAutomerger Merge Worker 	.version	= XTABLES_VERSION,
436*a71a9546SAutomerger Merge Worker 	.size		= XT_ALIGN(sizeof(struct xt_rateest_match_info)),
437*a71a9546SAutomerger Merge Worker 	.userspacesize	= XT_ALIGN(offsetof(struct xt_rateest_match_info, est1)),
438*a71a9546SAutomerger Merge Worker 	.help		= rateest_help,
439*a71a9546SAutomerger Merge Worker 	.parse		= rateest_parse,
440*a71a9546SAutomerger Merge Worker 	.x6_fcheck	= rateest_final_check,
441*a71a9546SAutomerger Merge Worker 	.print		= rateest_print,
442*a71a9546SAutomerger Merge Worker 	.save		= rateest_save,
443*a71a9546SAutomerger Merge Worker 	.extra_opts	= rateest_opts,
444*a71a9546SAutomerger Merge Worker };
445*a71a9546SAutomerger Merge Worker 
_init(void)446*a71a9546SAutomerger Merge Worker void _init(void)
447*a71a9546SAutomerger Merge Worker {
448*a71a9546SAutomerger Merge Worker 	xtables_register_match(&rateest_mt_reg);
449*a71a9546SAutomerger Merge Worker }
450