Blame extensions/libxt_set.c

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