Blame bootstrap_ver/extensions/libxt_set.c

Packit Service fa126c
/* Copyright (C) 2000-2002 Joakim Axelsson <gozem@linux.nu>
Packit Service fa126c
 *                         Patrick Schaaf <bof@bof.de>
Packit Service fa126c
 *                         Martin Josefsson <gandalf@wlug.westbo.se>
Packit Service fa126c
 * Copyright (C) 2003-2010 Jozsef Kadlecsik <kadlec@blackhole.kfki.hu>
Packit Service fa126c
 *
Packit Service fa126c
 * This program is free software; you can redistribute it and/or modify
Packit Service fa126c
 * it under the terms of the GNU General Public License version 2 as
Packit Service fa126c
 * published by the Free Software Foundation.  
Packit Service fa126c
 */
Packit Service fa126c
Packit Service fa126c
/* Shared library add-on to iptables to add IP set matching. */
Packit Service fa126c
#include <stdbool.h>
Packit Service fa126c
#include <stdio.h>
Packit Service fa126c
#include <netdb.h>
Packit Service fa126c
#include <string.h>
Packit Service fa126c
#include <stdlib.h>
Packit Service fa126c
#include <getopt.h>
Packit Service fa126c
#include <ctype.h>
Packit Service fa126c
#include <errno.h>
Packit Service fa126c
Packit Service fa126c
#include <xtables.h>
Packit Service fa126c
#include <linux/netfilter/xt_set.h>
Packit Service fa126c
#include "libxt_set.h"
Packit Service fa126c
Packit Service fa126c
/* Revision 0 */
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_help_v0(void)
Packit Service fa126c
{
Packit Service fa126c
	printf("set match options:\n"
Packit Service fa126c
	       " [!] --match-set name flags\n"
Packit Service fa126c
	       "		 'name' is the set name from to match,\n" 
Packit Service fa126c
	       "		 'flags' are the comma separated list of\n"
Packit Service fa126c
	       "		 'src' and 'dst' specifications.\n");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static const struct option set_opts_v0[] = {
Packit Service fa126c
	{.name = "match-set", .has_arg = true, .val = '1'},
Packit Service fa126c
	{.name = "set",       .has_arg = true, .val = '2'},
Packit Service fa126c
	XT_GETOPT_TABLEEND,
Packit Service fa126c
};
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_check_v0(unsigned int flags)
Packit Service fa126c
{
Packit Service fa126c
	if (!flags)
Packit Service fa126c
		xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
			"You must specify `--match-set' with proper arguments");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static int
Packit Service fa126c
set_parse_v0(int c, char **argv, int invert, unsigned int *flags,
Packit Service fa126c
	     const void *entry, struct xt_entry_match **match)
Packit Service fa126c
{
Packit Service fa126c
	struct xt_set_info_match_v0 *myinfo =
Packit Service fa126c
		(struct xt_set_info_match_v0 *) (*match)->data;
Packit Service fa126c
	struct xt_set_info_v0 *info = &myinfo->match_set;
Packit Service fa126c
Packit Service fa126c
	switch (c) {
Packit Service fa126c
	case '2':
Packit Service fa126c
		fprintf(stderr,
Packit Service fa126c
			"--set option deprecated, please use --match-set\n");
Packit Service fa126c
		/* fall through */
Packit Service fa126c
	case '1':		/* --match-set <set> <flag>[,<flag> */
Packit Service fa126c
		if (info->u.flags[0])
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set can be specified only once");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->u.flags[0] |= IPSET_MATCH_INV;
Packit Service fa126c
Packit Service fa126c
		if (!argv[optind]
Packit Service fa126c
		    || argv[optind][0] == '-'
Packit Service fa126c
		    || argv[optind][0] == '!')
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set requires two args.");
Packit Service fa126c
Packit Service fa126c
		if (strlen(optarg) > IPSET_MAXNAMELEN - 1)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "setname `%s' too long, max %d characters.",
Packit Service fa126c
				      optarg, IPSET_MAXNAMELEN - 1);
Packit Service fa126c
Packit Service fa126c
		get_set_byname(optarg, (struct xt_set_info *)info);
Packit Service fa126c
		parse_dirs_v0(argv[optind], info);
Packit Service fa126c
		DEBUGP("parse: set index %u\n", info->index);
Packit Service fa126c
		optind++;
Packit Service fa126c
Packit Service fa126c
		*flags = 1;
Packit Service fa126c
		break;
Packit Service fa126c
	}
Packit Service fa126c
Packit Service fa126c
	return 1;
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
print_match_v0(const char *prefix, const struct xt_set_info_v0 *info)
Packit Service fa126c
{
Packit Service fa126c
	int i;
Packit Service fa126c
	char setname[IPSET_MAXNAMELEN];
Packit Service fa126c
Packit Service fa126c
	get_set_byid(setname, info->index);
Packit Service fa126c
	printf("%s %s %s",
Packit Service fa126c
	       (info->u.flags[0] & IPSET_MATCH_INV) ? " !" : "",
Packit Service fa126c
	       prefix,
Packit Service fa126c
	       setname); 
Packit Service fa126c
	for (i = 0; i < IPSET_DIM_MAX; i++) {
Packit Service fa126c
		if (!info->u.flags[i])
Packit Service fa126c
			break;		
Packit Service fa126c
		printf("%s%s",
Packit Service fa126c
		       i == 0 ? " " : ",",
Packit Service fa126c
		       info->u.flags[i] & IPSET_SRC ? "src" : "dst");
Packit Service fa126c
	}
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Prints out the matchinfo. */
Packit Service fa126c
static void
Packit Service fa126c
set_print_v0(const void *ip, const struct xt_entry_match *match, int numeric)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v0 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	print_match_v0("match-set", &info->match_set);
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_save_v0(const void *ip, const struct xt_entry_match *match)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v0 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	print_match_v0("--match-set", &info->match_set);
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Revision 1 */
Packit Service fa126c
static int
Packit Service fa126c
set_parse_v1(int c, char **argv, int invert, unsigned int *flags,
Packit Service fa126c
	     const void *entry, struct xt_entry_match **match)
Packit Service fa126c
{
Packit Service fa126c
	struct xt_set_info_match_v1 *myinfo =
Packit Service fa126c
		(struct xt_set_info_match_v1 *) (*match)->data;
Packit Service fa126c
	struct xt_set_info *info = &myinfo->match_set;
Packit Service fa126c
Packit Service fa126c
	switch (c) {
Packit Service fa126c
	case '2':
Packit Service fa126c
		fprintf(stderr,
Packit Service fa126c
			"--set option deprecated, please use --match-set\n");
Packit Service fa126c
		/* fall through */
Packit Service fa126c
	case '1':		/* --match-set <set> <flag>[,<flag> */
Packit Service fa126c
		if (info->dim)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set can be specified only once");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->flags |= IPSET_INV_MATCH;
Packit Service fa126c
Packit Service fa126c
		if (!argv[optind]
Packit Service fa126c
		    || argv[optind][0] == '-'
Packit Service fa126c
		    || argv[optind][0] == '!')
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set requires two args.");
Packit Service fa126c
Packit Service fa126c
		if (strlen(optarg) > IPSET_MAXNAMELEN - 1)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "setname `%s' too long, max %d characters.",
Packit Service fa126c
				      optarg, IPSET_MAXNAMELEN - 1);
Packit Service fa126c
Packit Service fa126c
		get_set_byname(optarg, info);
Packit Service fa126c
		parse_dirs(argv[optind], info);
Packit Service fa126c
		DEBUGP("parse: set index %u\n", info->index);
Packit Service fa126c
		optind++;
Packit Service fa126c
Packit Service fa126c
		*flags = 1;
Packit Service fa126c
		break;
Packit Service fa126c
	}
Packit Service fa126c
Packit Service fa126c
	return 1;
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
print_match(const char *prefix, const struct xt_set_info *info)
Packit Service fa126c
{
Packit Service fa126c
	int i;
Packit Service fa126c
	char setname[IPSET_MAXNAMELEN];
Packit Service fa126c
Packit Service fa126c
	get_set_byid(setname, info->index);
Packit Service fa126c
	printf("%s %s %s",
Packit Service fa126c
	       (info->flags & IPSET_INV_MATCH) ? " !" : "",
Packit Service fa126c
	       prefix,
Packit Service fa126c
	       setname); 
Packit Service fa126c
	for (i = 1; i <= info->dim; i++) {		
Packit Service fa126c
		printf("%s%s",
Packit Service fa126c
		       i == 1 ? " " : ",",
Packit Service fa126c
		       info->flags & (1 << i) ? "src" : "dst");
Packit Service fa126c
	}
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Prints out the matchinfo. */
Packit Service fa126c
static void
Packit Service fa126c
set_print_v1(const void *ip, const struct xt_entry_match *match, int numeric)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v1 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	print_match("match-set", &info->match_set);
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_save_v1(const void *ip, const struct xt_entry_match *match)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v1 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	print_match("--match-set", &info->match_set);
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Revision 2 */
Packit Service fa126c
static void
Packit Service fa126c
set_help_v2(void)
Packit Service fa126c
{
Packit Service fa126c
	printf("set match options:\n"
Packit Service fa126c
	       " [!] --match-set name flags [--return-nomatch]\n"
Packit Service fa126c
	       "		 'name' is the set name from to match,\n" 
Packit Service fa126c
	       "		 'flags' are the comma separated list of\n"
Packit Service fa126c
	       "		 'src' and 'dst' specifications.\n");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static const struct option set_opts_v2[] = {
Packit Service fa126c
	{.name = "match-set",		.has_arg = true,	.val = '1'},
Packit Service fa126c
	{.name = "set",			.has_arg = true,	.val = '2'},
Packit Service fa126c
	{.name = "return-nomatch",	.has_arg = false,	.val = '3'},
Packit Service fa126c
	XT_GETOPT_TABLEEND,
Packit Service fa126c
};
Packit Service fa126c
Packit Service fa126c
static int
Packit Service fa126c
set_parse_v2(int c, char **argv, int invert, unsigned int *flags,
Packit Service fa126c
	     const void *entry, struct xt_entry_match **match)
Packit Service fa126c
{
Packit Service fa126c
	struct xt_set_info_match_v1 *myinfo =
Packit Service fa126c
		(struct xt_set_info_match_v1 *) (*match)->data;
Packit Service fa126c
	struct xt_set_info *info = &myinfo->match_set;
Packit Service fa126c
Packit Service fa126c
	switch (c) {
Packit Service fa126c
	case '3':
Packit Service fa126c
		info->flags |= IPSET_RETURN_NOMATCH;
Packit Service fa126c
		break;
Packit Service fa126c
	case '2':
Packit Service fa126c
		fprintf(stderr,
Packit Service fa126c
			"--set option deprecated, please use --match-set\n");
Packit Service fa126c
		/* fall through */
Packit Service fa126c
	case '1':		/* --match-set <set> <flag>[,<flag> */
Packit Service fa126c
		if (info->dim)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set can be specified only once");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->flags |= IPSET_INV_MATCH;
Packit Service fa126c
Packit Service fa126c
		if (!argv[optind]
Packit Service fa126c
		    || argv[optind][0] == '-'
Packit Service fa126c
		    || argv[optind][0] == '!')
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set requires two args.");
Packit Service fa126c
Packit Service fa126c
		if (strlen(optarg) > IPSET_MAXNAMELEN - 1)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "setname `%s' too long, max %d characters.",
Packit Service fa126c
				      optarg, IPSET_MAXNAMELEN - 1);
Packit Service fa126c
Packit Service fa126c
		get_set_byname(optarg, info);
Packit Service fa126c
		parse_dirs(argv[optind], info);
Packit Service fa126c
		DEBUGP("parse: set index %u\n", info->index);
Packit Service fa126c
		optind++;
Packit Service fa126c
Packit Service fa126c
		*flags = 1;
Packit Service fa126c
		break;
Packit Service fa126c
	}
Packit Service fa126c
Packit Service fa126c
	return 1;
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Prints out the matchinfo. */
Packit Service fa126c
static void
Packit Service fa126c
set_print_v2(const void *ip, const struct xt_entry_match *match, int numeric)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v1 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	print_match("match-set", &info->match_set);
Packit Service fa126c
	if (info->match_set.flags & IPSET_RETURN_NOMATCH)
Packit Service fa126c
		printf(" return-nomatch");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_save_v2(const void *ip, const struct xt_entry_match *match)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v1 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	print_match("--match-set", &info->match_set);
Packit Service fa126c
	if (info->match_set.flags & IPSET_RETURN_NOMATCH)
Packit Service fa126c
		printf(" --return-nomatch");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Revision 3 */
Packit Service fa126c
static void
Packit Service fa126c
set_help_v3(void)
Packit Service fa126c
{
Packit Service fa126c
	printf("set match options:\n"
Packit Service fa126c
	       " [!] --match-set name flags [--return-nomatch]\n"
Packit Service fa126c
	       "   [! --update-counters] [! --update-subcounters]\n"
Packit Service fa126c
	       "   [[!] --packets-eq value | --packets-lt value | --packets-gt value\n"
Packit Service fa126c
	       "   [[!] --bytes-eq value | --bytes-lt value | --bytes-gt value\n"
Packit Service fa126c
	       "		 'name' is the set name from to match,\n" 
Packit Service fa126c
	       "		 'flags' are the comma separated list of\n"
Packit Service fa126c
	       "		 'src' and 'dst' specifications.\n");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static const struct option set_opts_v3[] = {
Packit Service fa126c
	{.name = "match-set",		.has_arg = true,	.val = '1'},
Packit Service fa126c
	{.name = "set",			.has_arg = true,	.val = '2'},
Packit Service fa126c
	{.name = "return-nomatch",	.has_arg = false,	.val = '3'},
Packit Service fa126c
	{.name = "update-counters",	.has_arg = false,	.val = '4'},
Packit Service fa126c
	{.name = "packets-eq",		.has_arg = true,	.val = '5'},
Packit Service fa126c
	{.name = "packets-lt",		.has_arg = true,	.val = '6'},
Packit Service fa126c
	{.name = "packets-gt",		.has_arg = true,	.val = '7'},
Packit Service fa126c
	{.name = "bytes-eq",		.has_arg = true,	.val = '8'},
Packit Service fa126c
	{.name = "bytes-lt",		.has_arg = true,	.val = '9'},
Packit Service fa126c
	{.name = "bytes-gt",		.has_arg = true,	.val = '0'},
Packit Service fa126c
	{.name = "update-subcounters",	.has_arg = false,	.val = 'a'},
Packit Service fa126c
	XT_GETOPT_TABLEEND,
Packit Service fa126c
};
Packit Service fa126c
Packit Service fa126c
static uint64_t
Packit Service fa126c
parse_counter(const char *opt)
Packit Service fa126c
{
Packit Service fa126c
	uintmax_t value;
Packit Service fa126c
Packit Service fa126c
	if (!xtables_strtoul(opt, NULL, &value, 0, UINT64_MAX))
Packit Service fa126c
		xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
			      "Cannot parse %s as a counter value\n",
Packit Service fa126c
			      opt);
Packit Service fa126c
	return (uint64_t)value;
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static int
Packit Service fa126c
set_parse_v3(int c, char **argv, int invert, unsigned int *flags,
Packit Service fa126c
	     const void *entry, struct xt_entry_match **match)
Packit Service fa126c
{
Packit Service fa126c
	struct xt_set_info_match_v3 *info =
Packit Service fa126c
		(struct xt_set_info_match_v3 *) (*match)->data;
Packit Service fa126c
Packit Service fa126c
	switch (c) {
Packit Service fa126c
	case 'a':
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->flags |= IPSET_FLAG_SKIP_SUBCOUNTER_UPDATE;
Packit Service fa126c
		break;
Packit Service fa126c
	case '0':
Packit Service fa126c
		if (info->bytes.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --bytes-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--bytes-gt option cannot be inverted\n");
Packit Service fa126c
		info->bytes.op = IPSET_COUNTER_GT;
Packit Service fa126c
		info->bytes.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '9':
Packit Service fa126c
		if (info->bytes.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --bytes-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--bytes-lt option cannot be inverted\n");
Packit Service fa126c
		info->bytes.op = IPSET_COUNTER_LT;
Packit Service fa126c
		info->bytes.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '8':
Packit Service fa126c
		if (info->bytes.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --bytes-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		info->bytes.op = invert ? IPSET_COUNTER_NE : IPSET_COUNTER_EQ;
Packit Service fa126c
		info->bytes.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '7':
Packit Service fa126c
		if (info->packets.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --packets-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--packets-gt option cannot be inverted\n");
Packit Service fa126c
		info->packets.op = IPSET_COUNTER_GT;
Packit Service fa126c
		info->packets.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '6':
Packit Service fa126c
		if (info->packets.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --packets-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--packets-lt option cannot be inverted\n");
Packit Service fa126c
		info->packets.op = IPSET_COUNTER_LT;
Packit Service fa126c
		info->packets.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '5':
Packit Service fa126c
		if (info->packets.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --packets-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		info->packets.op = invert ? IPSET_COUNTER_NE : IPSET_COUNTER_EQ;
Packit Service fa126c
		info->packets.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '4':
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->flags |= IPSET_FLAG_SKIP_COUNTER_UPDATE;
Packit Service fa126c
		break;
Packit Service fa126c
	case '3':
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--return-nomatch flag cannot be inverted\n");
Packit Service fa126c
		info->flags |= IPSET_FLAG_RETURN_NOMATCH;
Packit Service fa126c
		break;
Packit Service fa126c
	case '2':
Packit Service fa126c
		fprintf(stderr,
Packit Service fa126c
			"--set option deprecated, please use --match-set\n");
Packit Service fa126c
		/* fall through */
Packit Service fa126c
	case '1':		/* --match-set <set> <flag>[,<flag> */
Packit Service fa126c
		if (info->match_set.dim)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set can be specified only once");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->match_set.flags |= IPSET_INV_MATCH;
Packit Service fa126c
Packit Service fa126c
		if (!argv[optind]
Packit Service fa126c
		    || argv[optind][0] == '-'
Packit Service fa126c
		    || argv[optind][0] == '!')
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set requires two args.");
Packit Service fa126c
Packit Service fa126c
		if (strlen(optarg) > IPSET_MAXNAMELEN - 1)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "setname `%s' too long, max %d characters.",
Packit Service fa126c
				      optarg, IPSET_MAXNAMELEN - 1);
Packit Service fa126c
Packit Service fa126c
		get_set_byname(optarg, &info->match_set);
Packit Service fa126c
		parse_dirs(argv[optind], &info->match_set);
Packit Service fa126c
		DEBUGP("parse: set index %u\n", info->match_set.index);
Packit Service fa126c
		optind++;
Packit Service fa126c
Packit Service fa126c
		*flags = 1;
Packit Service fa126c
		break;
Packit Service fa126c
	}
Packit Service fa126c
Packit Service fa126c
	return 1;
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_printv3_counter(const struct ip_set_counter_match0 *c, const char *name,
Packit Service fa126c
		    const char *sep)
Packit Service fa126c
{
Packit Service fa126c
	switch (c->op) {
Packit Service fa126c
	case IPSET_COUNTER_EQ:
Packit Service fa126c
		printf(" %s%s-eq %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	case IPSET_COUNTER_NE:
Packit Service fa126c
		printf(" ! %s%s-eq %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	case IPSET_COUNTER_LT:
Packit Service fa126c
		printf(" %s%s-lt %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	case IPSET_COUNTER_GT:
Packit Service fa126c
		printf(" %s%s-gt %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	}
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_print_v3_matchinfo(const struct xt_set_info_match_v3 *info,
Packit Service fa126c
		       const char *opt, const char *sep)
Packit Service fa126c
{
Packit Service fa126c
	print_match(opt, &info->match_set);
Packit Service fa126c
	if (info->flags & IPSET_FLAG_RETURN_NOMATCH)
Packit Service fa126c
		printf(" %sreturn-nomatch", sep);
Packit Service fa126c
	if ((info->flags & IPSET_FLAG_SKIP_COUNTER_UPDATE))
Packit Service fa126c
		printf(" ! %supdate-counters", sep);
Packit Service fa126c
	if ((info->flags & IPSET_FLAG_SKIP_SUBCOUNTER_UPDATE))
Packit Service fa126c
		printf(" ! %supdate-subcounters", sep);
Packit Service fa126c
	set_printv3_counter(&info->packets, "packets", sep);
Packit Service fa126c
	set_printv3_counter(&info->bytes, "bytes", sep);
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Prints out the matchinfo. */
Packit Service fa126c
static void
Packit Service fa126c
set_print_v3(const void *ip, const struct xt_entry_match *match, int numeric)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v3 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	set_print_v3_matchinfo(info, "match-set", "");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_save_v3(const void *ip, const struct xt_entry_match *match)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v3 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	set_print_v3_matchinfo(info, "--match-set", "--");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Revision 4 */
Packit Service fa126c
static int
Packit Service fa126c
set_parse_v4(int c, char **argv, int invert, unsigned int *flags,
Packit Service fa126c
	     const void *entry, struct xt_entry_match **match)
Packit Service fa126c
{
Packit Service fa126c
	struct xt_set_info_match_v4 *info =
Packit Service fa126c
		(struct xt_set_info_match_v4 *) (*match)->data;
Packit Service fa126c
Packit Service fa126c
	switch (c) {
Packit Service fa126c
	case 'a':
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->flags |= IPSET_FLAG_SKIP_SUBCOUNTER_UPDATE;
Packit Service fa126c
		break;
Packit Service fa126c
	case '0':
Packit Service fa126c
		if (info->bytes.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --bytes-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--bytes-gt option cannot be inverted\n");
Packit Service fa126c
		info->bytes.op = IPSET_COUNTER_GT;
Packit Service fa126c
		info->bytes.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '9':
Packit Service fa126c
		if (info->bytes.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --bytes-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--bytes-lt option cannot be inverted\n");
Packit Service fa126c
		info->bytes.op = IPSET_COUNTER_LT;
Packit Service fa126c
		info->bytes.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '8':
Packit Service fa126c
		if (info->bytes.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --bytes-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		info->bytes.op = invert ? IPSET_COUNTER_NE : IPSET_COUNTER_EQ;
Packit Service fa126c
		info->bytes.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '7':
Packit Service fa126c
		if (info->packets.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --packets-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--packets-gt option cannot be inverted\n");
Packit Service fa126c
		info->packets.op = IPSET_COUNTER_GT;
Packit Service fa126c
		info->packets.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '6':
Packit Service fa126c
		if (info->packets.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --packets-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--packets-lt option cannot be inverted\n");
Packit Service fa126c
		info->packets.op = IPSET_COUNTER_LT;
Packit Service fa126c
		info->packets.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '5':
Packit Service fa126c
		if (info->packets.op != IPSET_COUNTER_NONE)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "only one of the --packets-[eq|lt|gt]"
Packit Service fa126c
				      " is allowed\n");
Packit Service fa126c
		info->packets.op = invert ? IPSET_COUNTER_NE : IPSET_COUNTER_EQ;
Packit Service fa126c
		info->packets.value = parse_counter(optarg);
Packit Service fa126c
		break;
Packit Service fa126c
	case '4':
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->flags |= IPSET_FLAG_SKIP_COUNTER_UPDATE;
Packit Service fa126c
		break;
Packit Service fa126c
	case '3':
Packit Service fa126c
		if (invert)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--return-nomatch flag cannot be inverted\n");
Packit Service fa126c
		info->flags |= IPSET_FLAG_RETURN_NOMATCH;
Packit Service fa126c
		break;
Packit Service fa126c
	case '2':
Packit Service fa126c
		fprintf(stderr,
Packit Service fa126c
			"--set option deprecated, please use --match-set\n");
Packit Service fa126c
		/* fall through */
Packit Service fa126c
	case '1':		/* --match-set <set> <flag>[,<flag> */
Packit Service fa126c
		if (info->match_set.dim)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set can be specified only once");
Packit Service fa126c
		if (invert)
Packit Service fa126c
			info->match_set.flags |= IPSET_INV_MATCH;
Packit Service fa126c
Packit Service fa126c
		if (!argv[optind]
Packit Service fa126c
		    || argv[optind][0] == '-'
Packit Service fa126c
		    || argv[optind][0] == '!')
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "--match-set requires two args.");
Packit Service fa126c
Packit Service fa126c
		if (strlen(optarg) > IPSET_MAXNAMELEN - 1)
Packit Service fa126c
			xtables_error(PARAMETER_PROBLEM,
Packit Service fa126c
				      "setname `%s' too long, max %d characters.",
Packit Service fa126c
				      optarg, IPSET_MAXNAMELEN - 1);
Packit Service fa126c
Packit Service fa126c
		get_set_byname(optarg, &info->match_set);
Packit Service fa126c
		parse_dirs(argv[optind], &info->match_set);
Packit Service fa126c
		DEBUGP("parse: set index %u\n", info->match_set.index);
Packit Service fa126c
		optind++;
Packit Service fa126c
Packit Service fa126c
		*flags = 1;
Packit Service fa126c
		break;
Packit Service fa126c
	}
Packit Service fa126c
Packit Service fa126c
	return 1;
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_printv4_counter(const struct ip_set_counter_match *c, const char *name,
Packit Service fa126c
		    const char *sep)
Packit Service fa126c
{
Packit Service fa126c
	switch (c->op) {
Packit Service fa126c
	case IPSET_COUNTER_EQ:
Packit Service fa126c
		printf(" %s%s-eq %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	case IPSET_COUNTER_NE:
Packit Service fa126c
		printf(" ! %s%s-eq %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	case IPSET_COUNTER_LT:
Packit Service fa126c
		printf(" %s%s-lt %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	case IPSET_COUNTER_GT:
Packit Service fa126c
		printf(" %s%s-gt %llu", sep, name, c->value);
Packit Service fa126c
		break;
Packit Service fa126c
	}
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_print_v4_matchinfo(const struct xt_set_info_match_v4 *info,
Packit Service fa126c
		       const char *opt, const char *sep)
Packit Service fa126c
{
Packit Service fa126c
	print_match(opt, &info->match_set);
Packit Service fa126c
	if (info->flags & IPSET_FLAG_RETURN_NOMATCH)
Packit Service fa126c
		printf(" %sreturn-nomatch", sep);
Packit Service fa126c
	if ((info->flags & IPSET_FLAG_SKIP_COUNTER_UPDATE))
Packit Service fa126c
		printf(" ! %supdate-counters", sep);
Packit Service fa126c
	if ((info->flags & IPSET_FLAG_SKIP_SUBCOUNTER_UPDATE))
Packit Service fa126c
		printf(" ! %supdate-subcounters", sep);
Packit Service fa126c
	set_printv4_counter(&info->packets, "packets", sep);
Packit Service fa126c
	set_printv4_counter(&info->bytes, "bytes", sep);
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
/* Prints out the matchinfo. */
Packit Service fa126c
static void
Packit Service fa126c
set_print_v4(const void *ip, const struct xt_entry_match *match, int numeric)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v4 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	set_print_v4_matchinfo(info, "match-set", "");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static void
Packit Service fa126c
set_save_v4(const void *ip, const struct xt_entry_match *match)
Packit Service fa126c
{
Packit Service fa126c
	const struct xt_set_info_match_v4 *info = (const void *)match->data;
Packit Service fa126c
Packit Service fa126c
	set_print_v4_matchinfo(info, "--match-set", "--");
Packit Service fa126c
}
Packit Service fa126c
Packit Service fa126c
static struct xtables_match set_mt_reg[] = {
Packit Service fa126c
	{
Packit Service fa126c
		.name		= "set",
Packit Service fa126c
		.revision	= 0,
Packit Service fa126c
		.version	= XTABLES_VERSION,
Packit Service fa126c
		.family		= NFPROTO_IPV4,
Packit Service fa126c
		.size		= XT_ALIGN(sizeof(struct xt_set_info_match_v0)),
Packit Service fa126c
		.userspacesize	= XT_ALIGN(sizeof(struct xt_set_info_match_v0)),
Packit Service fa126c
		.help		= set_help_v0,
Packit Service fa126c
		.parse		= set_parse_v0,
Packit Service fa126c
		.final_check	= set_check_v0,
Packit Service fa126c
		.print		= set_print_v0,
Packit Service fa126c
		.save		= set_save_v0,
Packit Service fa126c
		.extra_opts	= set_opts_v0,
Packit Service fa126c
	},
Packit Service fa126c
	{
Packit Service fa126c
		.name		= "set",
Packit Service fa126c
		.revision	= 1,
Packit Service fa126c
		.version	= XTABLES_VERSION,
Packit Service fa126c
		.family		= NFPROTO_UNSPEC,
Packit Service fa126c
		.size		= XT_ALIGN(sizeof(struct xt_set_info_match_v1)),
Packit Service fa126c
		.userspacesize	= XT_ALIGN(sizeof(struct xt_set_info_match_v1)),
Packit Service fa126c
		.help		= set_help_v0,
Packit Service fa126c
		.parse		= set_parse_v1,
Packit Service fa126c
		.final_check	= set_check_v0,
Packit Service fa126c
		.print		= set_print_v1,
Packit Service fa126c
		.save		= set_save_v1,
Packit Service fa126c
		.extra_opts	= set_opts_v0,
Packit Service fa126c
	},
Packit Service fa126c
	{
Packit Service fa126c
		.name		= "set",
Packit Service fa126c
		.revision	= 2,
Packit Service fa126c
		.version	= XTABLES_VERSION,
Packit Service fa126c
		.family		= NFPROTO_UNSPEC,
Packit Service fa126c
		.size		= XT_ALIGN(sizeof(struct xt_set_info_match_v1)),
Packit Service fa126c
		.userspacesize	= XT_ALIGN(sizeof(struct xt_set_info_match_v1)),
Packit Service fa126c
		.help		= set_help_v2,
Packit Service fa126c
		.parse		= set_parse_v2,
Packit Service fa126c
		.final_check	= set_check_v0,
Packit Service fa126c
		.print		= set_print_v2,
Packit Service fa126c
		.save		= set_save_v2,
Packit Service fa126c
		.extra_opts	= set_opts_v2,
Packit Service fa126c
	},
Packit Service fa126c
	{
Packit Service fa126c
		.name		= "set",
Packit Service fa126c
		.revision	= 3,
Packit Service fa126c
		.version	= XTABLES_VERSION,
Packit Service fa126c
		.family		= NFPROTO_UNSPEC,
Packit Service fa126c
		.size		= XT_ALIGN(sizeof(struct xt_set_info_match_v3)),
Packit Service fa126c
		.userspacesize	= XT_ALIGN(sizeof(struct xt_set_info_match_v3)),
Packit Service fa126c
		.help		= set_help_v3,
Packit Service fa126c
		.parse		= set_parse_v3,
Packit Service fa126c
		.final_check	= set_check_v0,
Packit Service fa126c
		.print		= set_print_v3,
Packit Service fa126c
		.save		= set_save_v3,
Packit Service fa126c
		.extra_opts	= set_opts_v3,
Packit Service fa126c
	},
Packit Service fa126c
	{
Packit Service fa126c
		.name		= "set",
Packit Service fa126c
		.revision	= 4,
Packit Service fa126c
		.version	= XTABLES_VERSION,
Packit Service fa126c
		.family		= NFPROTO_UNSPEC,
Packit Service fa126c
		.size		= XT_ALIGN(sizeof(struct xt_set_info_match_v4)),
Packit Service fa126c
		.userspacesize	= XT_ALIGN(sizeof(struct xt_set_info_match_v4)),
Packit Service fa126c
		.help		= set_help_v3,
Packit Service fa126c
		.parse		= set_parse_v4,
Packit Service fa126c
		.final_check	= set_check_v0,
Packit Service fa126c
		.print		= set_print_v4,
Packit Service fa126c
		.save		= set_save_v4,
Packit Service fa126c
		.extra_opts	= set_opts_v3,
Packit Service fa126c
	},
Packit Service fa126c
};
Packit Service fa126c
Packit Service fa126c
void _init(void)
Packit Service fa126c
{
Packit Service fa126c
	xtables_register_matches(set_mt_reg, ARRAY_SIZE(set_mt_reg));
Packit Service fa126c
}