Blame extensions/libxt_policy.c

Packit Service d1fe03
/*
Packit Service d1fe03
 * Copyright (c) 2005-2013 Patrick McHardy <kaber@trash.net>
Packit Service d1fe03
 */
Packit Service d1fe03
Packit Service d1fe03
#include <stdbool.h>
Packit Service d1fe03
#include <stdint.h>
Packit Service d1fe03
#include <stdio.h>
Packit Service d1fe03
#include <string.h>
Packit Service d1fe03
#include <netdb.h>
Packit Service d1fe03
#include <xtables.h>
Packit Service d1fe03
#include <linux/netfilter/xt_policy.h>
Packit Service d1fe03
Packit Service d1fe03
enum {
Packit Service d1fe03
	O_DIRECTION = 0,
Packit Service d1fe03
	O_POLICY,
Packit Service d1fe03
	O_STRICT,
Packit Service d1fe03
	O_REQID,
Packit Service d1fe03
	O_SPI,
Packit Service d1fe03
	O_PROTO,
Packit Service d1fe03
	O_MODE,
Packit Service d1fe03
	O_TUNNELSRC,
Packit Service d1fe03
	O_TUNNELDST,
Packit Service d1fe03
	O_NEXT,
Packit Service d1fe03
	F_STRICT = 1 << O_STRICT,
Packit Service d1fe03
};
Packit Service d1fe03
Packit Service d1fe03
static void policy_help(void)
Packit Service d1fe03
{
Packit Service d1fe03
	printf(
Packit Service d1fe03
"policy match options:\n"
Packit Service d1fe03
"  --dir in|out			match policy applied during decapsulation/\n"
Packit Service d1fe03
"				policy to be applied during encapsulation\n"
Packit Service d1fe03
"  --pol none|ipsec		match policy\n"
Packit Service d1fe03
"  --strict 			match entire policy instead of single element\n"
Packit Service d1fe03
"				at any position\n"
Packit Service d1fe03
"These options may be used repeatedly, to describe policy elements:\n"
Packit Service d1fe03
"[!] --reqid reqid		match reqid\n"
Packit Service d1fe03
"[!] --spi spi			match SPI\n"
Packit Service d1fe03
"[!] --proto proto		match protocol (ah/esp/ipcomp)\n"
Packit Service d1fe03
"[!] --mode mode 		match mode (transport/tunnel)\n"
Packit Service d1fe03
"[!] --tunnel-src addr/mask	match tunnel source\n"
Packit Service d1fe03
"[!] --tunnel-dst addr/mask	match tunnel destination\n"
Packit Service d1fe03
"  --next 			begin next element in policy\n");
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static const struct xt_option_entry policy_opts[] = {
Packit Service d1fe03
	{.name = "dir", .id = O_DIRECTION, .type = XTTYPE_STRING},
Packit Service d1fe03
	{.name = "pol", .id = O_POLICY, .type = XTTYPE_STRING},
Packit Service d1fe03
	{.name = "strict", .id = O_STRICT, .type = XTTYPE_NONE},
Packit Service d1fe03
	{.name = "reqid", .id = O_REQID, .type = XTTYPE_UINT32,
Packit Service d1fe03
	 .flags = XTOPT_MULTI | XTOPT_INVERT},
Packit Service d1fe03
	{.name = "spi", .id = O_SPI, .type = XTTYPE_UINT32,
Packit Service d1fe03
	 .flags = XTOPT_MULTI | XTOPT_INVERT},
Packit Service d1fe03
	{.name = "tunnel-src", .id = O_TUNNELSRC, .type = XTTYPE_HOSTMASK,
Packit Service d1fe03
	 .flags = XTOPT_MULTI | XTOPT_INVERT},
Packit Service d1fe03
	{.name = "tunnel-dst", .id = O_TUNNELDST, .type = XTTYPE_HOSTMASK,
Packit Service d1fe03
	 .flags = XTOPT_MULTI | XTOPT_INVERT},
Packit Service d1fe03
	{.name = "proto", .id = O_PROTO, .type = XTTYPE_PROTOCOL,
Packit Service d1fe03
	 .flags = XTOPT_MULTI | XTOPT_INVERT},
Packit Service d1fe03
	{.name = "mode", .id = O_MODE, .type = XTTYPE_STRING,
Packit Service d1fe03
	 .flags = XTOPT_MULTI | XTOPT_INVERT},
Packit Service d1fe03
	{.name = "next", .id = O_NEXT, .type = XTTYPE_NONE,
Packit Service d1fe03
	 .flags = XTOPT_MULTI, .also = F_STRICT},
Packit Service d1fe03
	XTOPT_TABLEEND,
Packit Service d1fe03
};
Packit Service d1fe03
Packit Service d1fe03
static int parse_direction(const char *s)
Packit Service d1fe03
{
Packit Service d1fe03
	if (strcmp(s, "in") == 0)
Packit Service d1fe03
		return XT_POLICY_MATCH_IN;
Packit Service d1fe03
	if (strcmp(s, "out") == 0)
Packit Service d1fe03
		return XT_POLICY_MATCH_OUT;
Packit Service d1fe03
	xtables_error(PARAMETER_PROBLEM, "policy_match: invalid dir \"%s\"", s);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int parse_policy(const char *s)
Packit Service d1fe03
{
Packit Service d1fe03
	if (strcmp(s, "none") == 0)
Packit Service d1fe03
		return XT_POLICY_MATCH_NONE;
Packit Service d1fe03
	if (strcmp(s, "ipsec") == 0)
Packit Service d1fe03
		return 0;
Packit Service d1fe03
	xtables_error(PARAMETER_PROBLEM, "policy match: invalid policy \"%s\"", s);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int parse_mode(const char *s)
Packit Service d1fe03
{
Packit Service d1fe03
	if (strcmp(s, "transport") == 0)
Packit Service d1fe03
		return XT_POLICY_MODE_TRANSPORT;
Packit Service d1fe03
	if (strcmp(s, "tunnel") == 0)
Packit Service d1fe03
		return XT_POLICY_MODE_TUNNEL;
Packit Service d1fe03
	xtables_error(PARAMETER_PROBLEM, "policy match: invalid mode \"%s\"", s);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void policy_parse(struct xt_option_call *cb)
Packit Service d1fe03
{
Packit Service d1fe03
	struct xt_policy_info *info = cb->data;
Packit Service d1fe03
	struct xt_policy_elem *e = &info->pol[info->len];
Packit Service d1fe03
Packit Service d1fe03
	xtables_option_parse(cb);
Packit Service d1fe03
	switch (cb->entry->id) {
Packit Service d1fe03
	case O_DIRECTION:
Packit Service d1fe03
		info->flags |= parse_direction(cb->arg);
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_POLICY:
Packit Service d1fe03
		info->flags |= parse_policy(cb->arg);
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_STRICT:
Packit Service d1fe03
		info->flags |= XT_POLICY_MATCH_STRICT;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_REQID:
Packit Service d1fe03
		if (e->match.reqid)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: double --reqid option");
Packit Service d1fe03
		e->match.reqid = 1;
Packit Service d1fe03
		e->invert.reqid = cb->invert;
Packit Service d1fe03
		e->reqid = cb->val.u32;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_SPI:
Packit Service d1fe03
		if (e->match.spi)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: double --spi option");
Packit Service d1fe03
		e->match.spi = 1;
Packit Service d1fe03
		e->invert.spi = cb->invert;
Packit Service d1fe03
		e->spi = cb->val.u32;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_TUNNELSRC:
Packit Service d1fe03
		if (e->match.saddr)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: double --tunnel-src option");
Packit Service d1fe03
Packit Service d1fe03
		e->match.saddr = 1;
Packit Service d1fe03
		e->invert.saddr = cb->invert;
Packit Service d1fe03
		memcpy(&e->saddr, &cb->val.haddr, sizeof(cb->val.haddr));
Packit Service d1fe03
		memcpy(&e->smask, &cb->val.hmask, sizeof(cb->val.hmask));
Packit Service d1fe03
                break;
Packit Service d1fe03
	case O_TUNNELDST:
Packit Service d1fe03
		if (e->match.daddr)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: double --tunnel-dst option");
Packit Service d1fe03
		e->match.daddr = 1;
Packit Service d1fe03
		e->invert.daddr = cb->invert;
Packit Service d1fe03
		memcpy(&e->daddr, &cb->val.haddr, sizeof(cb->val.haddr));
Packit Service d1fe03
		memcpy(&e->dmask, &cb->val.hmask, sizeof(cb->val.hmask));
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_PROTO:
Packit Service d1fe03
		if (e->match.proto)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: double --proto option");
Packit Service d1fe03
		e->proto = cb->val.protocol;
Packit Service d1fe03
		if (e->proto != IPPROTO_AH && e->proto != IPPROTO_ESP &&
Packit Service d1fe03
		    e->proto != IPPROTO_COMP)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: protocol must be ah/esp/ipcomp");
Packit Service d1fe03
		e->match.proto = 1;
Packit Service d1fe03
		e->invert.proto = cb->invert;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_MODE:
Packit Service d1fe03
		if (e->match.mode)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: double --mode option");
Packit Service d1fe03
		e->match.mode = 1;
Packit Service d1fe03
		e->invert.mode = cb->invert;
Packit Service d1fe03
		e->mode = parse_mode(cb->arg);
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_NEXT:
Packit Service d1fe03
		if (++info->len == XT_POLICY_MAX_ELEM)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: maximum policy depth reached");
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void policy_check(struct xt_fcheck_call *cb)
Packit Service d1fe03
{
Packit Service d1fe03
	struct xt_policy_info *info = cb->data;
Packit Service d1fe03
	const struct xt_policy_elem *e;
Packit Service d1fe03
	int i;
Packit Service d1fe03
Packit Service d1fe03
	/*
Packit Service d1fe03
	 * The old "no parameters given" check is carried out
Packit Service d1fe03
	 * by testing for --dir.
Packit Service d1fe03
	 */
Packit Service d1fe03
	if (!(info->flags & (XT_POLICY_MATCH_IN | XT_POLICY_MATCH_OUT)))
Packit Service d1fe03
		xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
		           "policy match: neither --dir in nor --dir out specified");
Packit Service d1fe03
Packit Service d1fe03
	if (info->flags & XT_POLICY_MATCH_NONE) {
Packit Service d1fe03
		if (info->flags & XT_POLICY_MATCH_STRICT)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: policy none but --strict given");
Packit Service d1fe03
Packit Service d1fe03
		if (info->len != 0)
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: policy none but policy given");
Packit Service d1fe03
	} else
Packit Service d1fe03
		info->len++;	/* increase len by 1, no --next after last element */
Packit Service d1fe03
Packit Service d1fe03
	/*
Packit Service d1fe03
	 * This is already represented with O_NEXT requiring F_STRICT in the
Packit Service d1fe03
	 * options table, but will keep this code as a comment for reference.
Packit Service d1fe03
	 *
Packit Service d1fe03
	if (!(info->flags & XT_POLICY_MATCH_STRICT) && info->len > 1)
Packit Service d1fe03
		xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
		           "policy match: multiple elements but no --strict");
Packit Service d1fe03
	 */
Packit Service d1fe03
Packit Service d1fe03
	for (i = 0; i < info->len; i++) {
Packit Service d1fe03
		e = &info->pol[i];
Packit Service d1fe03
Packit Service d1fe03
		if (info->flags & XT_POLICY_MATCH_STRICT &&
Packit Service d1fe03
		    !(e->match.reqid || e->match.spi || e->match.saddr ||
Packit Service d1fe03
		      e->match.daddr || e->match.proto || e->match.mode))
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
				"policy match: empty policy element %u. "
Packit Service d1fe03
				"--strict is in effect, but at least one of "
Packit Service d1fe03
				"reqid, spi, tunnel-src, tunnel-dst, proto or "
Packit Service d1fe03
				"mode is required.", i);
Packit Service d1fe03
Packit Service d1fe03
		if ((e->match.saddr || e->match.daddr)
Packit Service d1fe03
		    && ((e->mode == XT_POLICY_MODE_TUNNEL && e->invert.mode) ||
Packit Service d1fe03
		        (e->mode == XT_POLICY_MODE_TRANSPORT && !e->invert.mode)))
Packit Service d1fe03
			xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			           "policy match: --tunnel-src/--tunnel-dst "
Packit Service d1fe03
			           "is only valid in tunnel mode");
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void print_mode(const char *prefix, uint8_t mode, int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	printf(" %smode ", prefix);
Packit Service d1fe03
Packit Service d1fe03
	switch (mode) {
Packit Service d1fe03
	case XT_POLICY_MODE_TRANSPORT:
Packit Service d1fe03
		printf("transport");
Packit Service d1fe03
		break;
Packit Service d1fe03
	case XT_POLICY_MODE_TUNNEL:
Packit Service d1fe03
		printf("tunnel");
Packit Service d1fe03
		break;
Packit Service d1fe03
	default:
Packit Service d1fe03
		printf("???");
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void print_proto(const char *prefix, uint8_t proto, int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct protoent *p = NULL;
Packit Service d1fe03
Packit Service d1fe03
	printf(" %sproto ", prefix);
Packit Service d1fe03
	if (!numeric)
Packit Service d1fe03
		p = getprotobynumber(proto);
Packit Service d1fe03
	if (p != NULL)
Packit Service d1fe03
		printf("%s", p->p_name);
Packit Service d1fe03
	else
Packit Service d1fe03
		printf("%u", proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
#define PRINT_INVERT(x)		\
Packit Service d1fe03
do {				\
Packit Service d1fe03
	if (x)			\
Packit Service d1fe03
		printf(" !");	\
Packit Service d1fe03
} while(0)
Packit Service d1fe03
Packit Service d1fe03
static void print_entry(const char *prefix, const struct xt_policy_elem *e,
Packit Service d1fe03
                        bool numeric, uint8_t family)
Packit Service d1fe03
{
Packit Service d1fe03
	if (e->match.reqid) {
Packit Service d1fe03
		PRINT_INVERT(e->invert.reqid);
Packit Service d1fe03
		printf(" %sreqid %u", prefix, e->reqid);
Packit Service d1fe03
	}
Packit Service d1fe03
	if (e->match.spi) {
Packit Service d1fe03
		PRINT_INVERT(e->invert.spi);
Packit Service d1fe03
		printf(" %sspi 0x%x", prefix, e->spi);
Packit Service d1fe03
	}
Packit Service d1fe03
	if (e->match.proto) {
Packit Service d1fe03
		PRINT_INVERT(e->invert.proto);
Packit Service d1fe03
		print_proto(prefix, e->proto, numeric);
Packit Service d1fe03
	}
Packit Service d1fe03
	if (e->match.mode) {
Packit Service d1fe03
		PRINT_INVERT(e->invert.mode);
Packit Service d1fe03
		print_mode(prefix, e->mode, numeric);
Packit Service d1fe03
	}
Packit Service d1fe03
	if (e->match.daddr) {
Packit Service d1fe03
		PRINT_INVERT(e->invert.daddr);
Packit Service d1fe03
		if (family == NFPROTO_IPV6)
Packit Service d1fe03
			printf(" %stunnel-dst %s%s", prefix,
Packit Service d1fe03
			       xtables_ip6addr_to_numeric(&e->daddr.a6),
Packit Service d1fe03
			       xtables_ip6mask_to_numeric(&e->dmask.a6));
Packit Service d1fe03
		else
Packit Service d1fe03
			printf(" %stunnel-dst %s%s", prefix,
Packit Service d1fe03
			       xtables_ipaddr_to_numeric(&e->daddr.a4),
Packit Service d1fe03
			       xtables_ipmask_to_numeric(&e->dmask.a4));
Packit Service d1fe03
	}
Packit Service d1fe03
	if (e->match.saddr) {
Packit Service d1fe03
		PRINT_INVERT(e->invert.saddr);
Packit Service d1fe03
		if (family == NFPROTO_IPV6)
Packit Service d1fe03
			printf(" %stunnel-src %s%s", prefix,
Packit Service d1fe03
			       xtables_ip6addr_to_numeric(&e->saddr.a6),
Packit Service d1fe03
			       xtables_ip6mask_to_numeric(&e->smask.a6));
Packit Service d1fe03
		else
Packit Service d1fe03
			printf(" %stunnel-src %s%s", prefix,
Packit Service d1fe03
			       xtables_ipaddr_to_numeric(&e->saddr.a4),
Packit Service d1fe03
			       xtables_ipmask_to_numeric(&e->smask.a4));
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void print_flags(const char *prefix, const struct xt_policy_info *info)
Packit Service d1fe03
{
Packit Service d1fe03
	if (info->flags & XT_POLICY_MATCH_IN)
Packit Service d1fe03
		printf(" %sdir in", prefix);
Packit Service d1fe03
	else
Packit Service d1fe03
		printf(" %sdir out", prefix);
Packit Service d1fe03
Packit Service d1fe03
	if (info->flags & XT_POLICY_MATCH_NONE)
Packit Service d1fe03
		printf(" %spol none", prefix);
Packit Service d1fe03
	else
Packit Service d1fe03
		printf(" %spol ipsec", prefix);
Packit Service d1fe03
Packit Service d1fe03
	if (info->flags & XT_POLICY_MATCH_STRICT)
Packit Service d1fe03
		printf(" %sstrict", prefix);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void policy4_print(const void *ip, const struct xt_entry_match *match,
Packit Service d1fe03
                          int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_policy_info *info = (void *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	printf(" policy match");
Packit Service d1fe03
	print_flags("", info);
Packit Service d1fe03
	for (i = 0; i < info->len; i++) {
Packit Service d1fe03
		if (info->len > 1)
Packit Service d1fe03
			printf(" [%u]", i);
Packit Service d1fe03
		print_entry("", &info->pol[i], numeric, NFPROTO_IPV4);
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void policy6_print(const void *ip, const struct xt_entry_match *match,
Packit Service d1fe03
                          int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_policy_info *info = (void *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	printf(" policy match");
Packit Service d1fe03
	print_flags("", info);
Packit Service d1fe03
	for (i = 0; i < info->len; i++) {
Packit Service d1fe03
		if (info->len > 1)
Packit Service d1fe03
			printf(" [%u]", i);
Packit Service d1fe03
		print_entry("", &info->pol[i], numeric, NFPROTO_IPV6);
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void policy4_save(const void *ip, const struct xt_entry_match *match)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_policy_info *info = (void *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	print_flags("--", info);
Packit Service d1fe03
	for (i = 0; i < info->len; i++) {
Packit Service d1fe03
		print_entry("--", &info->pol[i], false, NFPROTO_IPV4);
Packit Service d1fe03
		if (i + 1 < info->len)
Packit Service d1fe03
			printf(" --next");
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void policy6_save(const void *ip, const struct xt_entry_match *match)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_policy_info *info = (void *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	print_flags("--", info);
Packit Service d1fe03
	for (i = 0; i < info->len; i++) {
Packit Service d1fe03
		print_entry("--", &info->pol[i], false, NFPROTO_IPV6);
Packit Service d1fe03
		if (i + 1 < info->len)
Packit Service d1fe03
			printf(" --next");
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int policy_xlate(struct xt_xlate *xl,
Packit Service d1fe03
			const struct xt_xlate_mt_params *params)
Packit Service d1fe03
{
Packit Service d1fe03
	static const unsigned int allowed = XT_POLICY_MATCH_STRICT |
Packit Service d1fe03
					    XT_POLICY_MATCH_NONE |
Packit Service d1fe03
					    XT_POLICY_MATCH_IN;
Packit Service d1fe03
	static const struct xt_policy_elem empty;
Packit Service d1fe03
	const struct xt_policy_info *info = (const void *)params->match->data;
Packit Service d1fe03
Packit Service d1fe03
	if ((info->flags & ~allowed) || info->len > 1)
Packit Service d1fe03
		return 0;
Packit Service d1fe03
Packit Service d1fe03
	if (memcmp(&info->pol[0], &empty, sizeof(empty)))
Packit Service d1fe03
		return 0;
Packit Service d1fe03
Packit Service d1fe03
	xt_xlate_add(xl, "meta secpath ");
Packit Service d1fe03
Packit Service d1fe03
	if (info->flags & XT_POLICY_MATCH_NONE)
Packit Service d1fe03
		xt_xlate_add(xl, "missing");
Packit Service d1fe03
	else
Packit Service d1fe03
		xt_xlate_add(xl, "exists");
Packit Service d1fe03
Packit Service d1fe03
	return 1;
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static struct xtables_match policy_mt_reg[] = {
Packit Service d1fe03
	{
Packit Service d1fe03
		.name          = "policy",
Packit Service d1fe03
		.version       = XTABLES_VERSION,
Packit Service d1fe03
		.family        = NFPROTO_IPV4,
Packit Service d1fe03
		.size          = XT_ALIGN(sizeof(struct xt_policy_info)),
Packit Service d1fe03
		.userspacesize = XT_ALIGN(sizeof(struct xt_policy_info)),
Packit Service d1fe03
		.help          = policy_help,
Packit Service d1fe03
		.x6_parse      = policy_parse,
Packit Service d1fe03
		.x6_fcheck     = policy_check,
Packit Service d1fe03
		.print         = policy4_print,
Packit Service d1fe03
		.save          = policy4_save,
Packit Service d1fe03
		.x6_options    = policy_opts,
Packit Service d1fe03
		.xlate         = policy_xlate,
Packit Service d1fe03
	},
Packit Service d1fe03
	{
Packit Service d1fe03
		.name          = "policy",
Packit Service d1fe03
		.version       = XTABLES_VERSION,
Packit Service d1fe03
		.family        = NFPROTO_IPV6,
Packit Service d1fe03
		.size          = XT_ALIGN(sizeof(struct xt_policy_info)),
Packit Service d1fe03
		.userspacesize = XT_ALIGN(sizeof(struct xt_policy_info)),
Packit Service d1fe03
		.help          = policy_help,
Packit Service d1fe03
		.x6_parse      = policy_parse,
Packit Service d1fe03
		.x6_fcheck     = policy_check,
Packit Service d1fe03
		.print         = policy6_print,
Packit Service d1fe03
		.save          = policy6_save,
Packit Service d1fe03
		.x6_options    = policy_opts,
Packit Service d1fe03
		.xlate         = policy_xlate,
Packit Service d1fe03
	},
Packit Service d1fe03
};
Packit Service d1fe03
Packit Service d1fe03
void _init(void)
Packit Service d1fe03
{
Packit Service d1fe03
	xtables_register_matches(policy_mt_reg, ARRAY_SIZE(policy_mt_reg));
Packit Service d1fe03
}