blob: f97d07ec118ed8e91156c75e757d4888b493f123 [file] [log] [blame]
Bart De Schuymerff587202005-02-08 20:02:28 +00001/* ebt_among
2 *
3 * Authors:
4 * Grzegorz Borowiak <grzes@gnu.univ.gda.pl>
5 *
6 * August, 2003
7 */
8
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +00009#include <stdio.h>
10#include <string.h>
11#include <stdlib.h>
12#include <getopt.h>
13#include <ctype.h>
Bart De Schuymere1c315a2006-12-22 18:08:41 +000014#include <unistd.h>
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000015#include "../include/ebtables_u.h"
Bart De Schuymer57633b12010-12-18 16:15:49 +000016#include <netinet/ether.h>
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000017#include "../include/ethernetdb.h"
18#include <linux/if_ether.h>
19#include <linux/netfilter_bridge/ebt_among.h>
Bart De Schuymer8069b5a2006-08-17 10:18:02 +000020#include <sys/mman.h>
21#include <sys/stat.h>
22#include <fcntl.h>
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000023
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000024#define AMONG_DST '1'
25#define AMONG_SRC '2'
Bart De Schuymer8069b5a2006-08-17 10:18:02 +000026#define AMONG_DST_F '3'
27#define AMONG_SRC_F '4'
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000028
gborowiak6c6d7312003-09-16 19:26:38 +000029static struct option opts[] = {
30 {"among-dst", required_argument, 0, AMONG_DST},
31 {"among-src", required_argument, 0, AMONG_SRC},
Bart De Schuymer8069b5a2006-08-17 10:18:02 +000032 {"among-dst-file", required_argument, 0, AMONG_DST_F},
33 {"among-src-file", required_argument, 0, AMONG_SRC_F},
gborowiak6c6d7312003-09-16 19:26:38 +000034 {0}
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000035};
36
37#ifdef DEBUG
38static void hexdump(const void *mem, int howmany)
39{
40 printf("\n");
41 const unsigned char *p = mem;
42 int i;
Bart De Schuymerf8c97432003-09-27 17:39:09 +000043
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000044 for (i = 0; i < howmany; i++) {
45 if (i % 32 == 0) {
46 printf("\n%04x: ", i);
47 }
gborowiak6c6d7312003-09-16 19:26:38 +000048 printf("%2.2x%c", p[i], ". "[i % 4 == 3]);
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000049 }
50 printf("\n");
51}
gborowiak6c6d7312003-09-16 19:26:38 +000052#endif /* DEBUG */
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000053
54static void print_help()
55{
56 printf(
57"`among' options:\n"
Bart De Schuymer8069b5a2006-08-17 10:18:02 +000058"--among-dst [!] list : matches if ether dst is in list\n"
59"--among-src [!] list : matches if ether src is in list\n"
60"--among-dst-file [!] file : obtain dst list from file\n"
61"--among-src-file [!] file : obtain src list from file\n"
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000062"list has form:\n"
gborowiak6c6d7312003-09-16 19:26:38 +000063" xx:xx:xx:xx:xx:xx[=ip.ip.ip.ip],yy:yy:yy:yy:yy:yy[=ip.ip.ip.ip]"
64",...,zz:zz:zz:zz:zz:zz[=ip.ip.ip.ip][,]\n"
gborowiakc50ce6a2003-09-07 13:16:26 +000065"Things in brackets are optional.\n"
Bart De Schuymerf8c97432003-09-27 17:39:09 +000066"If you want to allow two (or more) IP addresses to one MAC address, you\n"
Bart De Schuymer7cc696b2003-10-12 12:34:10 +000067"can specify two (or more) pairs with the same MAC, e.g.\n"
gborowiakc50ce6a2003-09-07 13:16:26 +000068" 00:00:00:fa:eb:fe=153.19.120.250,00:00:00:fa:eb:fe=192.168.0.1\n"
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000069 );
70}
Bart De Schuymer1da19992008-03-04 19:14:29 +000071static int old_size;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000072
73static void init(struct ebt_entry_match *match)
74{
gborowiak6c6d7312003-09-16 19:26:38 +000075 struct ebt_among_info *amonginfo =
76 (struct ebt_among_info *) match->data;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000077
78 memset(amonginfo, 0, sizeof(struct ebt_among_info));
Bart De Schuymer1da19992008-03-04 19:14:29 +000079 old_size = sizeof(struct ebt_among_info);
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000080}
81
gborowiakc50ce6a2003-09-07 13:16:26 +000082static struct ebt_mac_wormhash *new_wormhash(int n)
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000083{
gborowiak6c6d7312003-09-16 19:26:38 +000084 int size =
85 sizeof(struct ebt_mac_wormhash) +
86 n * sizeof(struct ebt_mac_wormhash_tuple);
87 struct ebt_mac_wormhash *result =
88 (struct ebt_mac_wormhash *) malloc(size);
Bart De Schuymerf8c97432003-09-27 17:39:09 +000089
Bart De Schuymerff587202005-02-08 20:02:28 +000090 if (!result)
91 ebt_print_memory();
gborowiakc50ce6a2003-09-07 13:16:26 +000092 memset(result, 0, size);
93 result->poolsize = n;
94 return result;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +000095}
96
gborowiak6c6d7312003-09-16 19:26:38 +000097static void copy_wormhash(struct ebt_mac_wormhash *d,
98 const struct ebt_mac_wormhash *s)
gborowiakc50ce6a2003-09-07 13:16:26 +000099{
100 int dpoolsize = d->poolsize;
101 int dsize, ssize, amount;
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000102
gborowiakc50ce6a2003-09-07 13:16:26 +0000103 dsize = ebt_mac_wormhash_size(d);
104 ssize = ebt_mac_wormhash_size(s);
105 amount = dsize < ssize ? dsize : ssize;
106 memcpy(d, s, amount);
107 d->poolsize = dpoolsize;
108}
109
110/* Returns:
111 * -1 when '\0' reached
112 * -2 when `n' bytes read and no delimiter found
113 * 0 when no less than `n' bytes read and delimiter found
114 * if `destbuf' is not NULL, it is filled by read bytes and ended with '\0'
115 * *pp is set on the first byte not copied to `destbuf'
116 */
gborowiak6c6d7312003-09-16 19:26:38 +0000117static int read_until(const char **pp, const char *delimiters,
118 char *destbuf, int n)
gborowiakc50ce6a2003-09-07 13:16:26 +0000119{
120 int count = 0;
121 int ret = 0;
122 char c;
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000123
gborowiakc50ce6a2003-09-07 13:16:26 +0000124 while (1) {
125 c = **pp;
126 if (!c) {
127 ret = -1;
128 break;
129 }
130 if (strchr(delimiters, c)) {
131 ret = 0;
132 break;
133 }
134 if (count == n) {
135 ret = -2;
136 break;
137 }
gborowiak6c6d7312003-09-16 19:26:38 +0000138 if (destbuf)
139 destbuf[count++] = c;
gborowiakc50ce6a2003-09-07 13:16:26 +0000140 (*pp)++;
141 }
gborowiak6c6d7312003-09-16 19:26:38 +0000142 if (destbuf)
143 destbuf[count] = 0;
gborowiakc50ce6a2003-09-07 13:16:26 +0000144 return ret;
145}
146
gborowiak6c6d7312003-09-16 19:26:38 +0000147static int fcmp(const void *va, const void *vb) {
148 const struct ebt_mac_wormhash_tuple *a = va;
149 const struct ebt_mac_wormhash_tuple *b = vb;
150 int ca = ((const unsigned char*)a->cmp)[7];
151 int cb = ((const unsigned char*)b->cmp)[7];
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000152
gborowiak6c6d7312003-09-16 19:26:38 +0000153 return ca - cb;
154}
155
156static void index_table(struct ebt_mac_wormhash *wh)
157{
158 int ipool, itable;
159 int c;
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000160
gborowiak6c6d7312003-09-16 19:26:38 +0000161 for (itable = 0; itable <= 256; itable++) {
162 wh->table[itable] = wh->poolsize;
163 }
164 ipool = 0;
165 itable = 0;
166 while (1) {
167 wh->table[itable] = ipool;
168 c = ((const unsigned char*)wh->pool[ipool].cmp)[7];
169 if (itable <= c) {
170 itable++;
171 } else {
172 ipool++;
173 }
174 if (ipool > wh->poolsize)
175 break;
176 }
177}
178
gborowiakc50ce6a2003-09-07 13:16:26 +0000179static struct ebt_mac_wormhash *create_wormhash(const char *arg)
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000180{
181 const char *pc = arg;
182 const char *anchor;
gborowiakc50ce6a2003-09-07 13:16:26 +0000183 char *endptr;
184 struct ebt_mac_wormhash *workcopy, *result, *h;
185 unsigned char mac[6];
186 unsigned char ip[4];
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000187 int nmacs = 0;
gborowiakc50ce6a2003-09-07 13:16:26 +0000188 int i;
189 char token[4];
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000190
gborowiakc50ce6a2003-09-07 13:16:26 +0000191 if (!(workcopy = new_wormhash(1024))) {
Bart De Schuymer64182a32004-01-21 20:39:54 +0000192 ebt_print_memory();
gborowiakc50ce6a2003-09-07 13:16:26 +0000193 }
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000194 while (1) {
gborowiakc50ce6a2003-09-07 13:16:26 +0000195 /* remember current position, we'll need it on error */
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000196 anchor = pc;
gborowiakc50ce6a2003-09-07 13:16:26 +0000197
gborowiak6c6d7312003-09-16 19:26:38 +0000198 /* collect MAC; all its bytes are followed by ':' (colon),
199 * except for the last one which can be followed by
200 * ',' (comma), '=' or '\0' */
gborowiakc50ce6a2003-09-07 13:16:26 +0000201 for (i = 0; i < 5; i++) {
gborowiak6c6d7312003-09-16 19:26:38 +0000202 if (read_until(&pc, ":", token, 2) < 0
203 || token[0] == 0) {
Bart De Schuymerff587202005-02-08 20:02:28 +0000204 ebt_print_error("MAC parse error: %.20s", anchor);
Bart De Schuymer691254b2011-06-23 18:26:45 +0000205 free(workcopy);
Bart De Schuymerff587202005-02-08 20:02:28 +0000206 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000207 }
208 mac[i] = strtol(token, &endptr, 16);
209 if (*endptr) {
Bart De Schuymerff587202005-02-08 20:02:28 +0000210 ebt_print_error("MAC parse error: %.20s", anchor);
Bart De Schuymer691254b2011-06-23 18:26:45 +0000211 free(workcopy);
Bart De Schuymerff587202005-02-08 20:02:28 +0000212 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000213 }
214 pc++;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000215 }
gborowiakc50ce6a2003-09-07 13:16:26 +0000216 if (read_until(&pc, "=,", token, 2) == -2 || token[0] == 0) {
Bart De Schuymer64182a32004-01-21 20:39:54 +0000217 ebt_print_error("MAC parse error: %.20s", anchor);
Bart De Schuymerff587202005-02-08 20:02:28 +0000218 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000219 }
220 mac[i] = strtol(token, &endptr, 16);
221 if (*endptr) {
Bart De Schuymer64182a32004-01-21 20:39:54 +0000222 ebt_print_error("MAC parse error: %.20s", anchor);
Bart De Schuymerff587202005-02-08 20:02:28 +0000223 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000224 }
225 if (*pc == '=') {
226 /* an IP follows the MAC; collect similarly to MAC */
227 pc++;
228 anchor = pc;
229 for (i = 0; i < 3; i++) {
Bart De Schuymerff587202005-02-08 20:02:28 +0000230 if (read_until(&pc, ".", token, 3) < 0 || token[0] == 0) {
231 ebt_print_error("IP parse error: %.20s", anchor);
232 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000233 }
234 ip[i] = strtol(token, &endptr, 10);
235 if (*endptr) {
Bart De Schuymerff587202005-02-08 20:02:28 +0000236 ebt_print_error("IP parse error: %.20s", anchor);
237 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000238 }
239 pc++;
240 }
Bart De Schuymerff587202005-02-08 20:02:28 +0000241 if (read_until(&pc, ",", token, 3) == -2 || token[0] == 0) {
242 ebt_print_error("IP parse error: %.20s", anchor);
243 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000244 }
245 ip[3] = strtol(token, &endptr, 10);
246 if (*endptr) {
Bart De Schuymerff587202005-02-08 20:02:28 +0000247 ebt_print_error("IP parse error: %.20s", anchor);
248 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000249 }
Bart De Schuymer3f451932011-08-11 18:35:48 +0000250 if (ip[0] == 0 && ip[1] == 0 && ip[2] == 0 && ip[3] == 0) {
Bart De Schuymer64182a32004-01-21 20:39:54 +0000251 ebt_print_error("Illegal IP 0.0.0.0");
Bart De Schuymerff587202005-02-08 20:02:28 +0000252 return NULL;
gborowiak6c6d7312003-09-16 19:26:38 +0000253 }
254 } else {
gborowiakc50ce6a2003-09-07 13:16:26 +0000255 /* no IP, we set it to 0.0.0.0 */
256 memset(ip, 0, 4);
257 }
gborowiak6c6d7312003-09-16 19:26:38 +0000258
gborowiakc50ce6a2003-09-07 13:16:26 +0000259 /* we have collected MAC and IP, so we add an entry */
gborowiak6c6d7312003-09-16 19:26:38 +0000260 memcpy(((char *) workcopy->pool[nmacs].cmp) + 2, mac, 6);
Bart De Schuymer3f451932011-08-11 18:35:48 +0000261 memcpy(&(workcopy->pool[nmacs].ip), ip, 4);
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000262 nmacs++;
gborowiak6c6d7312003-09-16 19:26:38 +0000263
gborowiakc50ce6a2003-09-07 13:16:26 +0000264 /* re-allocate memory if needed */
265 if (*pc && nmacs >= workcopy->poolsize) {
266 if (!(h = new_wormhash(nmacs * 2))) {
Bart De Schuymer64182a32004-01-21 20:39:54 +0000267 ebt_print_memory();
gborowiakc50ce6a2003-09-07 13:16:26 +0000268 }
269 copy_wormhash(h, workcopy);
270 free(workcopy);
271 workcopy = h;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000272 }
gborowiak6c6d7312003-09-16 19:26:38 +0000273
gborowiakc50ce6a2003-09-07 13:16:26 +0000274 /* check if end of string was reached */
275 if (!*pc) {
276 break;
277 }
gborowiak6c6d7312003-09-16 19:26:38 +0000278
279 /* now `pc' points to comma if we are here; */
280 /* increment this to the next char */
gborowiakc50ce6a2003-09-07 13:16:26 +0000281 /* but first assert :-> */
282 if (*pc != ',') {
Bart De Schuymer64182a32004-01-21 20:39:54 +0000283 ebt_print_error("Something went wrong; no comma...\n");
Bart De Schuymerff587202005-02-08 20:02:28 +0000284 return NULL;
gborowiakc50ce6a2003-09-07 13:16:26 +0000285 }
286 pc++;
gborowiak6c6d7312003-09-16 19:26:38 +0000287
288 /* again check if end of string was reached; */
289 /* we allow an ending comma */
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000290 if (!*pc) {
291 break;
292 }
293 }
gborowiakc50ce6a2003-09-07 13:16:26 +0000294 if (!(result = new_wormhash(nmacs))) {
Bart De Schuymer64182a32004-01-21 20:39:54 +0000295 ebt_print_memory();
gborowiakc50ce6a2003-09-07 13:16:26 +0000296 }
297 copy_wormhash(result, workcopy);
298 free(workcopy);
gborowiak6c6d7312003-09-16 19:26:38 +0000299 qsort(&result->pool, result->poolsize,
300 sizeof(struct ebt_mac_wormhash_tuple), fcmp);
301 index_table(result);
gborowiakc50ce6a2003-09-07 13:16:26 +0000302 return result;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000303}
304
305#define OPT_DST 0x01
306#define OPT_SRC 0x02
gborowiak6c6d7312003-09-16 19:26:38 +0000307static int parse(int c, char **argv, int argc,
308 const struct ebt_u_entry *entry, unsigned int *flags,
309 struct ebt_entry_match **match)
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000310{
gborowiak6c6d7312003-09-16 19:26:38 +0000311 struct ebt_among_info *info =
312 (struct ebt_among_info *) (*match)->data;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000313 struct ebt_mac_wormhash *wh;
gborowiakc50ce6a2003-09-07 13:16:26 +0000314 struct ebt_entry_match *h;
Bart De Schuymer1da19992008-03-04 19:14:29 +0000315 int new_size;
Bart De Schuymer3f451932011-08-11 18:35:48 +0000316 long flen = 0;
317 int fd = -1;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000318
319 switch (c) {
Bart De Schuymer8069b5a2006-08-17 10:18:02 +0000320 case AMONG_DST_F:
321 case AMONG_SRC_F:
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000322 case AMONG_DST:
323 case AMONG_SRC:
Bart De Schuymer8069b5a2006-08-17 10:18:02 +0000324 if (c == AMONG_DST || c == AMONG_DST_F) {
Bart De Schuymerff587202005-02-08 20:02:28 +0000325 ebt_check_option2(flags, OPT_DST);
326 } else {
327 ebt_check_option2(flags, OPT_SRC);
328 }
329 if (ebt_check_inverse2(optarg)) {
Bart De Schuymer8069b5a2006-08-17 10:18:02 +0000330 if (c == AMONG_DST || c == AMONG_DST_F)
gborowiakc50ce6a2003-09-07 13:16:26 +0000331 info->bitmask |= EBT_AMONG_DST_NEG;
332 else
333 info->bitmask |= EBT_AMONG_SRC_NEG;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000334 }
Bart De Schuymer8069b5a2006-08-17 10:18:02 +0000335 if (c == AMONG_DST_F || c == AMONG_SRC_F) {
336 struct stat stats;
337
338 if ((fd = open(optarg, O_RDONLY)) == -1)
339 ebt_print_error("Couldn't open file '%s'", optarg);
340 fstat(fd, &stats);
341 flen = stats.st_size;
342 /* use mmap because the file will probably be big */
343 optarg = mmap(0, flen, PROT_READ|PROT_WRITE, MAP_PRIVATE, fd, 0);
344 if (optarg == MAP_FAILED)
345 ebt_print_error("Couldn't map file to memory");
346 if (optarg[flen-1] != '\n')
347 ebt_print_error("File should end with a newline");
348 if (strchr(optarg, '\n') != optarg+flen-1)
349 ebt_print_error("File should only contain one line");
350 optarg[flen-1] = '\0';
351 if (ebt_errormsg[0] != '\0') {
352 munmap(argv, flen);
353 close(fd);
354 exit(-1);
355 }
356 }
Bart De Schuymerff587202005-02-08 20:02:28 +0000357 wh = create_wormhash(optarg);
358 if (ebt_errormsg[0] != '\0')
359 break;
360
Bart De Schuymer1da19992008-03-04 19:14:29 +0000361 new_size = old_size+ebt_mac_wormhash_size(wh);
362 h = malloc(sizeof(struct ebt_entry_match)+EBT_ALIGN(new_size));
Bart De Schuymerff587202005-02-08 20:02:28 +0000363 if (!h)
364 ebt_print_memory();
Bart De Schuymer1da19992008-03-04 19:14:29 +0000365 memcpy(h, *match, old_size+sizeof(struct ebt_entry_match));
366 memcpy((char *)h+old_size+sizeof(struct ebt_entry_match), wh,
367 ebt_mac_wormhash_size(wh));
368 h->match_size = EBT_ALIGN(new_size);
gborowiak6c6d7312003-09-16 19:26:38 +0000369 info = (struct ebt_among_info *) h->data;
Bart De Schuymera64fd6c2010-04-22 06:57:57 +0000370 if (c == AMONG_DST || c == AMONG_DST_F) {
Bart De Schuymer1da19992008-03-04 19:14:29 +0000371 info->wh_dst_ofs = old_size;
gborowiakc50ce6a2003-09-07 13:16:26 +0000372 } else {
Bart De Schuymer1da19992008-03-04 19:14:29 +0000373 info->wh_src_ofs = old_size;
gborowiakc50ce6a2003-09-07 13:16:26 +0000374 }
Bart De Schuymer1da19992008-03-04 19:14:29 +0000375 old_size = new_size;
gborowiakc50ce6a2003-09-07 13:16:26 +0000376 free(*match);
377 *match = h;
378 free(wh);
Bart De Schuymer8069b5a2006-08-17 10:18:02 +0000379 if (c == AMONG_DST_F || c == AMONG_SRC_F) {
380 munmap(argv, flen);
381 close(fd);
382 }
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000383 break;
384 default:
385 return 0;
386 }
387 return 1;
388}
389
390static void final_check(const struct ebt_u_entry *entry,
gborowiak6c6d7312003-09-16 19:26:38 +0000391 const struct ebt_entry_match *match,
392 const char *name, unsigned int hookmask,
393 unsigned int time)
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000394{
395}
396
gborowiak6c6d7312003-09-16 19:26:38 +0000397#ifdef DEBUG
398static void wormhash_debug(const struct ebt_mac_wormhash *wh)
399{
400 int i;
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000401
gborowiak6c6d7312003-09-16 19:26:38 +0000402 printf("poolsize: %d\n", wh->poolsize);
403 for (i = 0; i <= 256; i++) {
404 printf("%02x ", wh->table[i]);
405 if (i % 16 == 15) {
406 printf("\n");
407 }
408 }
409 printf("\n");
410}
411#endif /* DEBUG */
412
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000413static void wormhash_printout(const struct ebt_mac_wormhash *wh)
414{
415 int i;
gborowiakc50ce6a2003-09-07 13:16:26 +0000416 unsigned char *ip;
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000417
gborowiak6c6d7312003-09-16 19:26:38 +0000418 for (i = 0; i < wh->poolsize; i++) {
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000419 const struct ebt_mac_wormhash_tuple *p;
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000420
gborowiak6c6d7312003-09-16 19:26:38 +0000421 p = (const struct ebt_mac_wormhash_tuple *)(&wh->pool[i]);
Bart De Schuymer510c9ce2006-01-23 18:50:54 +0000422 ebt_print_mac(((const unsigned char *) &p->cmp[0]) + 2);
gborowiak6c6d7312003-09-16 19:26:38 +0000423 if (p->ip) {
424 ip = (unsigned char *) &p->ip;
425 printf("=%u.%u.%u.%u", ip[0], ip[1], ip[2], ip[3]);
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000426 }
gborowiak6c6d7312003-09-16 19:26:38 +0000427 printf(",");
428 }
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000429 printf(" ");
430}
431
432static void print(const struct ebt_u_entry *entry,
gborowiak6c6d7312003-09-16 19:26:38 +0000433 const struct ebt_entry_match *match)
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000434{
gborowiakc50ce6a2003-09-07 13:16:26 +0000435 struct ebt_among_info *info = (struct ebt_among_info *)match->data;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000436
gborowiakc50ce6a2003-09-07 13:16:26 +0000437 if (info->wh_dst_ofs) {
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000438 printf("--among-dst ");
gborowiakc50ce6a2003-09-07 13:16:26 +0000439 if (info->bitmask && EBT_AMONG_DST_NEG) {
440 printf("! ");
441 }
442 wormhash_printout(ebt_among_wh_dst(info));
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000443 }
gborowiakc50ce6a2003-09-07 13:16:26 +0000444 if (info->wh_src_ofs) {
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000445 printf("--among-src ");
gborowiakc50ce6a2003-09-07 13:16:26 +0000446 if (info->bitmask && EBT_AMONG_SRC_NEG) {
447 printf("! ");
448 }
449 wormhash_printout(ebt_among_wh_src(info));
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000450 }
451}
452
gborowiak6c6d7312003-09-16 19:26:38 +0000453static int compare_wh(const struct ebt_mac_wormhash *aw,
454 const struct ebt_mac_wormhash *bw)
gborowiakc50ce6a2003-09-07 13:16:26 +0000455{
456 int as, bs;
Bart De Schuymerf8c97432003-09-27 17:39:09 +0000457
gborowiakc50ce6a2003-09-07 13:16:26 +0000458 as = ebt_mac_wormhash_size(aw);
459 bs = ebt_mac_wormhash_size(bw);
460 if (as != bs)
461 return 0;
462 if (as && memcmp(aw, bw, as))
463 return 0;
464 return 1;
465}
466
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000467static int compare(const struct ebt_entry_match *m1,
gborowiak6c6d7312003-09-16 19:26:38 +0000468 const struct ebt_entry_match *m2)
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000469{
gborowiak6c6d7312003-09-16 19:26:38 +0000470 struct ebt_among_info *a = (struct ebt_among_info *) m1->data;
471 struct ebt_among_info *b = (struct ebt_among_info *) m2->data;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000472
gborowiak6c6d7312003-09-16 19:26:38 +0000473 if (!compare_wh(ebt_among_wh_dst(a), ebt_among_wh_dst(b)))
474 return 0;
475 if (!compare_wh(ebt_among_wh_src(a), ebt_among_wh_src(b)))
476 return 0;
477 if (a->bitmask != b->bitmask)
478 return 0;
gborowiakc50ce6a2003-09-07 13:16:26 +0000479 return 1;
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000480}
481
gborowiak6c6d7312003-09-16 19:26:38 +0000482static struct ebt_u_match among_match = {
Bart De Schuymer138bf062009-06-21 12:38:05 +0000483 .name = "among",
gborowiak6c6d7312003-09-16 19:26:38 +0000484 .size = sizeof(struct ebt_among_info),
485 .help = print_help,
486 .init = init,
487 .parse = parse,
488 .final_check = final_check,
489 .print = print,
490 .compare = compare,
491 .extra_ops = opts,
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000492};
493
Bart De Schuymer64182a32004-01-21 20:39:54 +0000494void _init(void)
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000495{
Bart De Schuymer8339ff12004-01-14 20:05:27 +0000496 ebt_register_match(&among_match);
Bart De Schuymer9cc9bfa2003-09-02 22:43:25 +0000497}