Git init
[external/iptables.git] / extensions / libxt_statistic.c
1 #include <stdio.h>
2 #include <netdb.h>
3 #include <string.h>
4 #include <stdlib.h>
5 #include <stddef.h>
6 #include <getopt.h>
7
8 #include <xtables.h>
9 #include <linux/netfilter/xt_statistic.h>
10
11 static void statistic_help(void)
12 {
13         printf(
14 "statistic match options:\n"
15 " --mode mode                    Match mode (random, nth)\n"
16 " random mode:\n"
17 " --probability p                Probability\n"
18 " nth mode:\n"
19 " --every n                      Match every nth packet\n"
20 " --packet p                     Initial counter value (0 <= p <= n-1, default 0)\n");
21 }
22
23 static const struct option statistic_opts[] = {
24         { "mode", 1, NULL, '1' },
25         { "probability", 1, NULL, '2' },
26         { "every", 1, NULL, '3' },
27         { "packet", 1, NULL, '4' },
28         { .name = NULL }
29 };
30
31 static struct xt_statistic_info *global_info;
32
33 static void statistic_mt_init(struct xt_entry_match *match)
34 {
35         global_info = (void *)match->data;
36 }
37
38 static int
39 statistic_parse(int c, char **argv, int invert, unsigned int *flags,
40                 const void *entry, struct xt_entry_match **match)
41 {
42         struct xt_statistic_info *info = (void *)(*match)->data;
43         unsigned int val;
44         double prob;
45
46         if (invert)
47                 info->flags |= XT_STATISTIC_INVERT;
48
49         switch (c) {
50         case '1':
51                 if (*flags & 0x1)
52                         xtables_error(PARAMETER_PROBLEM, "double --mode");
53                 if (!strcmp(optarg, "random"))
54                         info->mode = XT_STATISTIC_MODE_RANDOM;
55                 else if (!strcmp(optarg, "nth"))
56                         info->mode = XT_STATISTIC_MODE_NTH;
57                 else
58                         xtables_error(PARAMETER_PROBLEM, "Bad mode \"%s\"", optarg);
59                 *flags |= 0x1;
60                 break;
61         case '2':
62                 if (*flags & 0x2)
63                         xtables_error(PARAMETER_PROBLEM, "double --probability");
64                 prob = atof(optarg);
65                 if (prob < 0 || prob > 1)
66                         xtables_error(PARAMETER_PROBLEM,
67                                    "--probability must be between 0 and 1");
68                 info->u.random.probability = 0x80000000 * prob;
69                 *flags |= 0x2;
70                 break;
71         case '3':
72                 if (*flags & 0x4)
73                         xtables_error(PARAMETER_PROBLEM, "double --every");
74                 if (!xtables_strtoui(optarg, NULL, &val, 0, UINT32_MAX))
75                         xtables_error(PARAMETER_PROBLEM,
76                                    "cannot parse --every `%s'", optarg);
77                 info->u.nth.every = val;
78                 if (info->u.nth.every == 0)
79                         xtables_error(PARAMETER_PROBLEM, "--every cannot be 0");
80                 info->u.nth.every--;
81                 *flags |= 0x4;
82                 break;
83         case '4':
84                 if (*flags & 0x8)
85                         xtables_error(PARAMETER_PROBLEM, "double --packet");
86                 if (!xtables_strtoui(optarg, NULL, &val, 0, UINT32_MAX))
87                         xtables_error(PARAMETER_PROBLEM,
88                                    "cannot parse --packet `%s'", optarg);
89                 info->u.nth.packet = val;
90                 *flags |= 0x8;
91                 break;
92         default:
93                 return 0;
94         }
95         return 1;
96 }
97
98 static void statistic_check(unsigned int flags)
99 {
100         if (!(flags & 0x1))
101                 xtables_error(PARAMETER_PROBLEM, "no mode specified");
102         if ((flags & 0x2) && (flags & (0x4 | 0x8)))
103                 xtables_error(PARAMETER_PROBLEM,
104                            "both nth and random parameters given");
105         if (flags & 0x2 && global_info->mode != XT_STATISTIC_MODE_RANDOM)
106                 xtables_error(PARAMETER_PROBLEM,
107                            "--probability can only be used in random mode");
108         if (flags & 0x4 && global_info->mode != XT_STATISTIC_MODE_NTH)
109                 xtables_error(PARAMETER_PROBLEM,
110                            "--every can only be used in nth mode");
111         if (flags & 0x8 && global_info->mode != XT_STATISTIC_MODE_NTH)
112                 xtables_error(PARAMETER_PROBLEM,
113                            "--packet can only be used in nth mode");
114         if ((flags & 0x8) && !(flags & 0x4))
115                 xtables_error(PARAMETER_PROBLEM,
116                            "--packet can only be used with --every");
117         /* at this point, info->u.nth.every have been decreased. */
118         if (global_info->u.nth.packet > global_info->u.nth.every)
119                 xtables_error(PARAMETER_PROBLEM,
120                           "the --packet p must be 0 <= p <= n-1");
121
122
123         global_info->u.nth.count = global_info->u.nth.every -
124                                    global_info->u.nth.packet;
125 }
126
127 static void print_match(const struct xt_statistic_info *info, char *prefix)
128 {
129         if (info->flags & XT_STATISTIC_INVERT)
130                 printf("! ");
131
132         switch (info->mode) {
133         case XT_STATISTIC_MODE_RANDOM:
134                 printf("%smode random %sprobability %f ", prefix, prefix,
135                        1.0 * info->u.random.probability / 0x80000000);
136                 break;
137         case XT_STATISTIC_MODE_NTH:
138                 printf("%smode nth %severy %u ", prefix, prefix,
139                        info->u.nth.every + 1);
140                 if (info->u.nth.packet)
141                         printf("%spacket %u ", prefix, info->u.nth.packet);
142                 break;
143         }
144 }
145
146 static void
147 statistic_print(const void *ip, const struct xt_entry_match *match, int numeric)
148 {
149         const struct xt_statistic_info *info = (const void *)match->data;
150
151         printf("statistic ");
152         print_match(info, "");
153 }
154
155 static void statistic_save(const void *ip, const struct xt_entry_match *match)
156 {
157         const struct xt_statistic_info *info = (const void *)match->data;
158
159         print_match(info, "--");
160 }
161
162 static struct xtables_match statistic_match = {
163         .family         = NFPROTO_UNSPEC,
164         .name           = "statistic",
165         .version        = XTABLES_VERSION,
166         .size           = XT_ALIGN(sizeof(struct xt_statistic_info)),
167         .userspacesize  = offsetof(struct xt_statistic_info, u.nth.count),
168         .init           = statistic_mt_init,
169         .help           = statistic_help,
170         .parse          = statistic_parse,
171         .final_check    = statistic_check,
172         .print          = statistic_print,
173         .save           = statistic_save,
174         .extra_opts     = statistic_opts,
175 };
176
177 void _init(void)
178 {
179         xtables_register_match(&statistic_match);
180 }