Blame extensions/libxt_multiport.c

Packit Service d1fe03
#include <stdio.h>
Packit Service d1fe03
#include <netdb.h>
Packit Service d1fe03
#include <string.h>
Packit Service d1fe03
#include <stdlib.h>
Packit Service d1fe03
#include <xtables.h>
Packit Service d1fe03
#include <limits.h> /* INT_MAX in ip_tables.h/ip6_tables.h */
Packit Service d1fe03
#include <linux/netfilter_ipv4/ip_tables.h>
Packit Service d1fe03
#include <linux/netfilter_ipv6/ip6_tables.h>
Packit Service d1fe03
#include <linux/netfilter/xt_multiport.h>
Packit Service d1fe03
Packit Service d1fe03
enum {
Packit Service d1fe03
	O_SOURCE_PORTS = 0,
Packit Service d1fe03
	O_DEST_PORTS,
Packit Service d1fe03
	O_SD_PORTS,
Packit Service d1fe03
	F_SOURCE_PORTS = 1 << O_SOURCE_PORTS,
Packit Service d1fe03
	F_DEST_PORTS   = 1 << O_DEST_PORTS,
Packit Service d1fe03
	F_SD_PORTS     = 1 << O_SD_PORTS,
Packit Service d1fe03
	F_ANY          = F_SOURCE_PORTS | F_DEST_PORTS | F_SD_PORTS,
Packit Service d1fe03
};
Packit Service d1fe03
Packit Service d1fe03
/* Function which prints out usage message. */
Packit Service d1fe03
static void multiport_help(void)
Packit Service d1fe03
{
Packit Service d1fe03
	printf(
Packit Service d1fe03
"multiport match options:\n"
Packit Service d1fe03
" --source-ports port[,port,port...]\n"
Packit Service d1fe03
" --sports ...\n"
Packit Service d1fe03
"				match source port(s)\n"
Packit Service d1fe03
" --destination-ports port[,port,port...]\n"
Packit Service d1fe03
" --dports ...\n"
Packit Service d1fe03
"				match destination port(s)\n"
Packit Service d1fe03
" --ports port[,port,port]\n"
Packit Service d1fe03
"				match both source and destination port(s)\n"
Packit Service d1fe03
" NOTE: this kernel does not support port ranges in multiport.\n");
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_help_v1(void)
Packit Service d1fe03
{
Packit Service d1fe03
	printf(
Packit Service d1fe03
"multiport match options:\n"
Packit Service d1fe03
"[!] --source-ports port[,port:port,port...]\n"
Packit Service d1fe03
" --sports ...\n"
Packit Service d1fe03
"				match source port(s)\n"
Packit Service d1fe03
"[!] --destination-ports port[,port:port,port...]\n"
Packit Service d1fe03
" --dports ...\n"
Packit Service d1fe03
"				match destination port(s)\n"
Packit Service d1fe03
"[!] --ports port[,port:port,port]\n"
Packit Service d1fe03
"				match both source and destination port(s)\n");
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static const struct xt_option_entry multiport_opts[] = {
Packit Service d1fe03
	{.name = "source-ports", .id = O_SOURCE_PORTS, .type = XTTYPE_STRING,
Packit Service d1fe03
	 .excl = F_ANY, .flags = XTOPT_INVERT},
Packit Service d1fe03
	{.name = "sports", .id = O_SOURCE_PORTS, .type = XTTYPE_STRING,
Packit Service d1fe03
	 .excl = F_ANY, .flags = XTOPT_INVERT},
Packit Service d1fe03
	{.name = "destination-ports", .id = O_DEST_PORTS,
Packit Service d1fe03
	 .type = XTTYPE_STRING, .excl = F_ANY, .flags = XTOPT_INVERT},
Packit Service d1fe03
	{.name = "dports", .id = O_DEST_PORTS, .type = XTTYPE_STRING,
Packit Service d1fe03
	 .excl = F_ANY, .flags = XTOPT_INVERT},
Packit Service d1fe03
	{.name = "ports", .id = O_SD_PORTS, .type = XTTYPE_STRING,
Packit Service d1fe03
	 .excl = F_ANY, .flags = XTOPT_INVERT},
Packit Service d1fe03
	XTOPT_TABLEEND,
Packit Service d1fe03
};
Packit Service d1fe03
Packit Service d1fe03
static const char *
Packit Service d1fe03
proto_to_name(uint8_t proto)
Packit Service d1fe03
{
Packit Service d1fe03
	switch (proto) {
Packit Service d1fe03
	case IPPROTO_TCP:
Packit Service d1fe03
		return "tcp";
Packit Service d1fe03
	case IPPROTO_UDP:
Packit Service d1fe03
		return "udp";
Packit Service d1fe03
	case IPPROTO_UDPLITE:
Packit Service d1fe03
		return "udplite";
Packit Service d1fe03
	case IPPROTO_SCTP:
Packit Service d1fe03
		return "sctp";
Packit Service d1fe03
	case IPPROTO_DCCP:
Packit Service d1fe03
		return "dccp";
Packit Service d1fe03
	default:
Packit Service d1fe03
		return NULL;
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static unsigned int
Packit Service d1fe03
parse_multi_ports(const char *portstring, uint16_t *ports, const char *proto)
Packit Service d1fe03
{
Packit Service d1fe03
	char *buffer, *cp, *next;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	buffer = strdup(portstring);
Packit Service d1fe03
	if (!buffer) xtables_error(OTHER_PROBLEM, "strdup failed");
Packit Service d1fe03
Packit Service d1fe03
	for (cp=buffer, i=0; cp && i
Packit Service d1fe03
	{
Packit Service d1fe03
		next=strchr(cp, ',');
Packit Service d1fe03
		if (next) *next++='\0';
Packit Service d1fe03
		ports[i] = xtables_parse_port(cp, proto);
Packit Service d1fe03
	}
Packit Service d1fe03
	if (cp) xtables_error(PARAMETER_PROBLEM, "too many ports specified");
Packit Service d1fe03
	free(buffer);
Packit Service d1fe03
	return i;
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void
Packit Service d1fe03
parse_multi_ports_v1(const char *portstring, 
Packit Service d1fe03
		     struct xt_multiport_v1 *multiinfo,
Packit Service d1fe03
		     const char *proto)
Packit Service d1fe03
{
Packit Service d1fe03
	char *buffer, *cp, *next, *range;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	buffer = strdup(portstring);
Packit Service d1fe03
	if (!buffer) xtables_error(OTHER_PROBLEM, "strdup failed");
Packit Service d1fe03
Packit Service d1fe03
	for (i=0; i
Packit Service d1fe03
		multiinfo->pflags[i] = 0;
Packit Service d1fe03
 
Packit Service d1fe03
	for (cp=buffer, i=0; cp && i
Packit Service d1fe03
		next=strchr(cp, ',');
Packit Service d1fe03
 		if (next) *next++='\0';
Packit Service d1fe03
		range = strchr(cp, ':');
Packit Service d1fe03
		if (range) {
Packit Service d1fe03
			if (i == XT_MULTI_PORTS-1)
Packit Service d1fe03
				xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
					   "too many ports specified");
Packit Service d1fe03
			*range++ = '\0';
Packit Service d1fe03
		}
Packit Service d1fe03
		multiinfo->ports[i] = xtables_parse_port(cp, proto);
Packit Service d1fe03
		if (range) {
Packit Service d1fe03
			multiinfo->pflags[i] = 1;
Packit Service d1fe03
			multiinfo->ports[++i] = xtables_parse_port(range, proto);
Packit Service d1fe03
			if (multiinfo->ports[i-1] >= multiinfo->ports[i])
Packit Service d1fe03
				xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
					   "invalid portrange specified");
Packit Service d1fe03
		}
Packit Service d1fe03
 	}
Packit Service d1fe03
	multiinfo->count = i;
Packit Service d1fe03
	if (cp) xtables_error(PARAMETER_PROBLEM, "too many ports specified");
Packit Service d1fe03
 	free(buffer);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static const char *
Packit Service d1fe03
check_proto(uint16_t pnum, uint8_t invflags)
Packit Service d1fe03
{
Packit Service d1fe03
	const char *proto;
Packit Service d1fe03
Packit Service d1fe03
	if (invflags & XT_INV_PROTO)
Packit Service d1fe03
		xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			   "multiport only works with TCP, UDP, UDPLITE, SCTP and DCCP");
Packit Service d1fe03
Packit Service d1fe03
	if ((proto = proto_to_name(pnum)) != NULL)
Packit Service d1fe03
		return proto;
Packit Service d1fe03
	else if (!pnum)
Packit Service d1fe03
		xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			   "multiport needs `-p tcp', `-p udp', `-p udplite', "
Packit Service d1fe03
			   "`-p sctp' or `-p dccp'");
Packit Service d1fe03
	else
Packit Service d1fe03
		xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			   "multiport only works with TCP, UDP, UDPLITE, SCTP and DCCP");
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void __multiport_parse(struct xt_option_call *cb, uint16_t pnum,
Packit Service d1fe03
			      uint8_t invflags)
Packit Service d1fe03
{
Packit Service d1fe03
	const char *proto;
Packit Service d1fe03
	struct xt_multiport *multiinfo = cb->data;
Packit Service d1fe03
Packit Service d1fe03
	xtables_option_parse(cb);
Packit Service d1fe03
	switch (cb->entry->id) {
Packit Service d1fe03
	case O_SOURCE_PORTS:
Packit Service d1fe03
		proto = check_proto(pnum, invflags);
Packit Service d1fe03
		multiinfo->count = parse_multi_ports(cb->arg,
Packit Service d1fe03
						     multiinfo->ports, proto);
Packit Service d1fe03
		multiinfo->flags = XT_MULTIPORT_SOURCE;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_DEST_PORTS:
Packit Service d1fe03
		proto = check_proto(pnum, invflags);
Packit Service d1fe03
		multiinfo->count = parse_multi_ports(cb->arg,
Packit Service d1fe03
						     multiinfo->ports, proto);
Packit Service d1fe03
		multiinfo->flags = XT_MULTIPORT_DESTINATION;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_SD_PORTS:
Packit Service d1fe03
		proto = check_proto(pnum, invflags);
Packit Service d1fe03
		multiinfo->count = parse_multi_ports(cb->arg,
Packit Service d1fe03
						     multiinfo->ports, proto);
Packit Service d1fe03
		multiinfo->flags = XT_MULTIPORT_EITHER;
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
	if (cb->invert)
Packit Service d1fe03
		xtables_error(PARAMETER_PROBLEM,
Packit Service d1fe03
			   "multiport.0 does not support invert");
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_parse(struct xt_option_call *cb)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ipt_entry *entry = cb->xt_entry;
Packit Service d1fe03
	return __multiport_parse(cb,
Packit Service d1fe03
	       entry->ip.proto, entry->ip.invflags);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_parse6(struct xt_option_call *cb)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ip6t_entry *entry = cb->xt_entry;
Packit Service d1fe03
	return __multiport_parse(cb,
Packit Service d1fe03
	       entry->ipv6.proto, entry->ipv6.invflags);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void __multiport_parse_v1(struct xt_option_call *cb, uint16_t pnum,
Packit Service d1fe03
				 uint8_t invflags)
Packit Service d1fe03
{
Packit Service d1fe03
	const char *proto;
Packit Service d1fe03
	struct xt_multiport_v1 *multiinfo = cb->data;
Packit Service d1fe03
Packit Service d1fe03
	xtables_option_parse(cb);
Packit Service d1fe03
	switch (cb->entry->id) {
Packit Service d1fe03
	case O_SOURCE_PORTS:
Packit Service d1fe03
		proto = check_proto(pnum, invflags);
Packit Service d1fe03
		parse_multi_ports_v1(cb->arg, multiinfo, proto);
Packit Service d1fe03
		multiinfo->flags = XT_MULTIPORT_SOURCE;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_DEST_PORTS:
Packit Service d1fe03
		proto = check_proto(pnum, invflags);
Packit Service d1fe03
		parse_multi_ports_v1(cb->arg, multiinfo, proto);
Packit Service d1fe03
		multiinfo->flags = XT_MULTIPORT_DESTINATION;
Packit Service d1fe03
		break;
Packit Service d1fe03
	case O_SD_PORTS:
Packit Service d1fe03
		proto = check_proto(pnum, invflags);
Packit Service d1fe03
		parse_multi_ports_v1(cb->arg, multiinfo, proto);
Packit Service d1fe03
		multiinfo->flags = XT_MULTIPORT_EITHER;
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
	if (cb->invert)
Packit Service d1fe03
		multiinfo->invert = 1;
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_parse_v1(struct xt_option_call *cb)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ipt_entry *entry = cb->xt_entry;
Packit Service d1fe03
	return __multiport_parse_v1(cb,
Packit Service d1fe03
	       entry->ip.proto, entry->ip.invflags);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_parse6_v1(struct xt_option_call *cb)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ip6t_entry *entry = cb->xt_entry;
Packit Service d1fe03
	return __multiport_parse_v1(cb,
Packit Service d1fe03
	       entry->ipv6.proto, entry->ipv6.invflags);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_check(struct xt_fcheck_call *cb)
Packit Service d1fe03
{
Packit Service d1fe03
	if (cb->xflags == 0)
Packit Service d1fe03
		xtables_error(PARAMETER_PROBLEM, "multiport expection an option");
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static const char *
Packit Service d1fe03
port_to_service(int port, uint8_t proto)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct servent *service;
Packit Service d1fe03
Packit Service d1fe03
	if ((service = getservbyport(htons(port), proto_to_name(proto))))
Packit Service d1fe03
		return service->s_name;
Packit Service d1fe03
Packit Service d1fe03
	return NULL;
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void
Packit Service d1fe03
print_port(uint16_t port, uint8_t protocol, int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const char *service;
Packit Service d1fe03
Packit Service d1fe03
	if (numeric || (service = port_to_service(port, protocol)) == NULL)
Packit Service d1fe03
		printf("%u", port);
Packit Service d1fe03
	else
Packit Service d1fe03
		printf("%s", service);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void
Packit Service d1fe03
__multiport_print(const struct xt_entry_match *match, int numeric,
Packit Service d1fe03
                  uint16_t proto)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_multiport *multiinfo
Packit Service d1fe03
		= (const struct xt_multiport *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	printf(" multiport ");
Packit Service d1fe03
Packit Service d1fe03
	switch (multiinfo->flags) {
Packit Service d1fe03
	case XT_MULTIPORT_SOURCE:
Packit Service d1fe03
		printf("sports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_DESTINATION:
Packit Service d1fe03
		printf("dports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_EITHER:
Packit Service d1fe03
		printf("ports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	default:
Packit Service d1fe03
		printf("ERROR ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
Packit Service d1fe03
	for (i=0; i < multiinfo->count; i++) {
Packit Service d1fe03
		printf("%s", i ? "," : "");
Packit Service d1fe03
		print_port(multiinfo->ports[i], proto, numeric);
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_print(const void *ip_void,
Packit Service d1fe03
                            const struct xt_entry_match *match, int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ipt_ip *ip = ip_void;
Packit Service d1fe03
	__multiport_print(match, numeric, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_print6(const void *ip_void,
Packit Service d1fe03
                             const struct xt_entry_match *match, int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ip6t_ip6 *ip = ip_void;
Packit Service d1fe03
	__multiport_print(match, numeric, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void __multiport_print_v1(const struct xt_entry_match *match,
Packit Service d1fe03
                                 int numeric, uint16_t proto)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_multiport_v1 *multiinfo
Packit Service d1fe03
		= (const struct xt_multiport_v1 *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	printf(" multiport ");
Packit Service d1fe03
Packit Service d1fe03
	switch (multiinfo->flags) {
Packit Service d1fe03
	case XT_MULTIPORT_SOURCE:
Packit Service d1fe03
		printf("sports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_DESTINATION:
Packit Service d1fe03
		printf("dports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_EITHER:
Packit Service d1fe03
		printf("ports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	default:
Packit Service d1fe03
		printf("ERROR ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
Packit Service d1fe03
	if (multiinfo->invert)
Packit Service d1fe03
		printf(" !");
Packit Service d1fe03
Packit Service d1fe03
	for (i=0; i < multiinfo->count; i++) {
Packit Service d1fe03
		printf("%s", i ? "," : "");
Packit Service d1fe03
		print_port(multiinfo->ports[i], proto, numeric);
Packit Service d1fe03
		if (multiinfo->pflags[i]) {
Packit Service d1fe03
			printf(":");
Packit Service d1fe03
			print_port(multiinfo->ports[++i], proto, numeric);
Packit Service d1fe03
		}
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_print_v1(const void *ip_void,
Packit Service d1fe03
                               const struct xt_entry_match *match, int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ipt_ip *ip = ip_void;
Packit Service d1fe03
	__multiport_print_v1(match, numeric, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_print6_v1(const void *ip_void,
Packit Service d1fe03
                                const struct xt_entry_match *match, int numeric)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ip6t_ip6 *ip = ip_void;
Packit Service d1fe03
	__multiport_print_v1(match, numeric, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void __multiport_save(const struct xt_entry_match *match,
Packit Service d1fe03
                             uint16_t proto)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_multiport *multiinfo
Packit Service d1fe03
		= (const struct xt_multiport *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	switch (multiinfo->flags) {
Packit Service d1fe03
	case XT_MULTIPORT_SOURCE:
Packit Service d1fe03
		printf(" --sports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_DESTINATION:
Packit Service d1fe03
		printf(" --dports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_EITHER:
Packit Service d1fe03
		printf(" --ports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
Packit Service d1fe03
	for (i=0; i < multiinfo->count; i++) {
Packit Service d1fe03
		printf("%s", i ? "," : "");
Packit Service d1fe03
		print_port(multiinfo->ports[i], proto, 1);
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_save(const void *ip_void,
Packit Service d1fe03
                           const struct xt_entry_match *match)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ipt_ip *ip = ip_void;
Packit Service d1fe03
	__multiport_save(match, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_save6(const void *ip_void,
Packit Service d1fe03
                            const struct xt_entry_match *match)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ip6t_ip6 *ip = ip_void;
Packit Service d1fe03
	__multiport_save(match, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void __multiport_save_v1(const struct xt_entry_match *match,
Packit Service d1fe03
                                uint16_t proto)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_multiport_v1 *multiinfo
Packit Service d1fe03
		= (const struct xt_multiport_v1 *)match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	if (multiinfo->invert)
Packit Service d1fe03
		printf(" !");
Packit Service d1fe03
Packit Service d1fe03
	switch (multiinfo->flags) {
Packit Service d1fe03
	case XT_MULTIPORT_SOURCE:
Packit Service d1fe03
		printf(" --sports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_DESTINATION:
Packit Service d1fe03
		printf(" --dports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
Packit Service d1fe03
	case XT_MULTIPORT_EITHER:
Packit Service d1fe03
		printf(" --ports ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	}
Packit Service d1fe03
Packit Service d1fe03
	for (i=0; i < multiinfo->count; i++) {
Packit Service d1fe03
		printf("%s", i ? "," : "");
Packit Service d1fe03
		print_port(multiinfo->ports[i], proto, 1);
Packit Service d1fe03
		if (multiinfo->pflags[i]) {
Packit Service d1fe03
			printf(":");
Packit Service d1fe03
			print_port(multiinfo->ports[++i], proto, 1);
Packit Service d1fe03
		}
Packit Service d1fe03
	}
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_save_v1(const void *ip_void,
Packit Service d1fe03
                              const struct xt_entry_match *match)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ipt_ip *ip = ip_void;
Packit Service d1fe03
	__multiport_save_v1(match, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static void multiport_save6_v1(const void *ip_void,
Packit Service d1fe03
                               const struct xt_entry_match *match)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct ip6t_ip6 *ip = ip_void;
Packit Service d1fe03
	__multiport_save_v1(match, ip->proto);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int __multiport_xlate(struct xt_xlate *xl,
Packit Service d1fe03
			     const struct xt_xlate_mt_params *params)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_multiport *multiinfo
Packit Service d1fe03
		= (const struct xt_multiport *)params->match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	switch (multiinfo->flags) {
Packit Service d1fe03
	case XT_MULTIPORT_SOURCE:
Packit Service d1fe03
		xt_xlate_add(xl, " sport ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	case XT_MULTIPORT_DESTINATION:
Packit Service d1fe03
		xt_xlate_add(xl, " dport ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	case XT_MULTIPORT_EITHER:
Packit Service d1fe03
		return 0;
Packit Service d1fe03
	}
Packit Service d1fe03
Packit Service d1fe03
	if (multiinfo->count > 1)
Packit Service d1fe03
		xt_xlate_add(xl, "{ ");
Packit Service d1fe03
Packit Service d1fe03
	for (i = 0; i < multiinfo->count; i++)
Packit Service d1fe03
		xt_xlate_add(xl, "%s%u", i ? "," : "", multiinfo->ports[i]);
Packit Service d1fe03
Packit Service d1fe03
	if (multiinfo->count > 1)
Packit Service d1fe03
		xt_xlate_add(xl, "}");
Packit Service d1fe03
Packit Service d1fe03
	return 1;
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int multiport_xlate(struct xt_xlate *xl,
Packit Service d1fe03
			   const struct xt_xlate_mt_params *params)
Packit Service d1fe03
{
Packit Service d1fe03
	uint8_t proto = ((const struct ipt_ip *)params->ip)->proto;
Packit Service d1fe03
Packit Service d1fe03
	xt_xlate_add(xl, "%s", proto_to_name(proto));
Packit Service d1fe03
	return __multiport_xlate(xl, params);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int multiport_xlate6(struct xt_xlate *xl,
Packit Service d1fe03
			    const struct xt_xlate_mt_params *params)
Packit Service d1fe03
{
Packit Service d1fe03
	uint8_t proto = ((const struct ip6t_ip6 *)params->ip)->proto;
Packit Service d1fe03
Packit Service d1fe03
	xt_xlate_add(xl, "%s", proto_to_name(proto));
Packit Service d1fe03
	return __multiport_xlate(xl, params);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int __multiport_xlate_v1(struct xt_xlate *xl,
Packit Service d1fe03
				const struct xt_xlate_mt_params *params)
Packit Service d1fe03
{
Packit Service d1fe03
	const struct xt_multiport_v1 *multiinfo =
Packit Service d1fe03
		(const struct xt_multiport_v1 *)params->match->data;
Packit Service d1fe03
	unsigned int i;
Packit Service d1fe03
Packit Service d1fe03
	switch (multiinfo->flags) {
Packit Service d1fe03
	case XT_MULTIPORT_SOURCE:
Packit Service d1fe03
		xt_xlate_add(xl, " sport ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	case XT_MULTIPORT_DESTINATION:
Packit Service d1fe03
		xt_xlate_add(xl, " dport ");
Packit Service d1fe03
		break;
Packit Service d1fe03
	case XT_MULTIPORT_EITHER:
Packit Service d1fe03
		return 0;
Packit Service d1fe03
	}
Packit Service d1fe03
Packit Service d1fe03
	if (multiinfo->invert)
Packit Service d1fe03
		xt_xlate_add(xl, "!= ");
Packit Service d1fe03
Packit Service d1fe03
	if (multiinfo->count > 2 ||
Packit Service d1fe03
	    (multiinfo->count > 1 && !multiinfo->pflags[0]))
Packit Service d1fe03
		xt_xlate_add(xl, "{ ");
Packit Service d1fe03
Packit Service d1fe03
	for (i = 0; i < multiinfo->count; i++) {
Packit Service d1fe03
		xt_xlate_add(xl, "%s%u", i ? "," : "", multiinfo->ports[i]);
Packit Service d1fe03
		if (multiinfo->pflags[i])
Packit Service d1fe03
			xt_xlate_add(xl, "-%u", multiinfo->ports[++i]);
Packit Service d1fe03
	}
Packit Service d1fe03
Packit Service d1fe03
	if (multiinfo->count > 2 ||
Packit Service d1fe03
	    (multiinfo->count > 1 && !multiinfo->pflags[0]))
Packit Service d1fe03
		xt_xlate_add(xl, "}");
Packit Service d1fe03
Packit Service d1fe03
	return 1;
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int multiport_xlate_v1(struct xt_xlate *xl,
Packit Service d1fe03
			      const struct xt_xlate_mt_params *params)
Packit Service d1fe03
{
Packit Service d1fe03
	uint8_t proto = ((const struct ipt_ip *)params->ip)->proto;
Packit Service d1fe03
Packit Service d1fe03
	xt_xlate_add(xl, "%s", proto_to_name(proto));
Packit Service d1fe03
	return __multiport_xlate_v1(xl, params);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static int multiport_xlate6_v1(struct xt_xlate *xl,
Packit Service d1fe03
			       const struct xt_xlate_mt_params *params)
Packit Service d1fe03
{
Packit Service d1fe03
	uint8_t proto = ((const struct ip6t_ip6 *)params->ip)->proto;
Packit Service d1fe03
Packit Service d1fe03
	xt_xlate_add(xl, "%s", proto_to_name(proto));
Packit Service d1fe03
	return __multiport_xlate_v1(xl, params);
Packit Service d1fe03
}
Packit Service d1fe03
Packit Service d1fe03
static struct xtables_match multiport_mt_reg[] = {
Packit Service d1fe03
	{
Packit Service d1fe03
		.family        = NFPROTO_IPV4,
Packit Service d1fe03
		.name          = "multiport",
Packit Service d1fe03
		.revision      = 0,
Packit Service d1fe03
		.version       = XTABLES_VERSION,
Packit Service d1fe03
		.size          = XT_ALIGN(sizeof(struct xt_multiport)),
Packit Service d1fe03
		.userspacesize = XT_ALIGN(sizeof(struct xt_multiport)),
Packit Service d1fe03
		.help          = multiport_help,
Packit Service d1fe03
		.x6_parse      = multiport_parse,
Packit Service d1fe03
		.x6_fcheck     = multiport_check,
Packit Service d1fe03
		.print         = multiport_print,
Packit Service d1fe03
		.save          = multiport_save,
Packit Service d1fe03
		.x6_options    = multiport_opts,
Packit Service d1fe03
		.xlate         = multiport_xlate,
Packit Service d1fe03
	},
Packit Service d1fe03
	{
Packit Service d1fe03
		.family        = NFPROTO_IPV6,
Packit Service d1fe03
		.name          = "multiport",
Packit Service d1fe03
		.revision      = 0,
Packit Service d1fe03
		.version       = XTABLES_VERSION,
Packit Service d1fe03
		.size          = XT_ALIGN(sizeof(struct xt_multiport)),
Packit Service d1fe03
		.userspacesize = XT_ALIGN(sizeof(struct xt_multiport)),
Packit Service d1fe03
		.help          = multiport_help,
Packit Service d1fe03
		.x6_parse      = multiport_parse6,
Packit Service d1fe03
		.x6_fcheck     = multiport_check,
Packit Service d1fe03
		.print         = multiport_print6,
Packit Service d1fe03
		.save          = multiport_save6,
Packit Service d1fe03
		.x6_options    = multiport_opts,
Packit Service d1fe03
		.xlate         = multiport_xlate6,
Packit Service d1fe03
	},
Packit Service d1fe03
	{
Packit Service d1fe03
		.family        = NFPROTO_IPV4,
Packit Service d1fe03
		.name          = "multiport",
Packit Service d1fe03
		.version       = XTABLES_VERSION,
Packit Service d1fe03
		.revision      = 1,
Packit Service d1fe03
		.size          = XT_ALIGN(sizeof(struct xt_multiport_v1)),
Packit Service d1fe03
		.userspacesize = XT_ALIGN(sizeof(struct xt_multiport_v1)),
Packit Service d1fe03
		.help          = multiport_help_v1,
Packit Service d1fe03
		.x6_parse      = multiport_parse_v1,
Packit Service d1fe03
		.x6_fcheck     = multiport_check,
Packit Service d1fe03
		.print         = multiport_print_v1,
Packit Service d1fe03
		.save          = multiport_save_v1,
Packit Service d1fe03
		.x6_options    = multiport_opts,
Packit Service d1fe03
		.xlate         = multiport_xlate_v1,
Packit Service d1fe03
	},
Packit Service d1fe03
	{
Packit Service d1fe03
		.family        = NFPROTO_IPV6,
Packit Service d1fe03
		.name          = "multiport",
Packit Service d1fe03
		.version       = XTABLES_VERSION,
Packit Service d1fe03
		.revision      = 1,
Packit Service d1fe03
		.size          = XT_ALIGN(sizeof(struct xt_multiport_v1)),
Packit Service d1fe03
		.userspacesize = XT_ALIGN(sizeof(struct xt_multiport_v1)),
Packit Service d1fe03
		.help          = multiport_help_v1,
Packit Service d1fe03
		.x6_parse      = multiport_parse6_v1,
Packit Service d1fe03
		.x6_fcheck     = multiport_check,
Packit Service d1fe03
		.print         = multiport_print6_v1,
Packit Service d1fe03
		.save          = multiport_save6_v1,
Packit Service d1fe03
		.x6_options    = multiport_opts,
Packit Service d1fe03
		.xlate         = multiport_xlate6_v1,
Packit Service d1fe03
	},
Packit Service d1fe03
};
Packit Service d1fe03
Packit Service d1fe03
void
Packit Service d1fe03
_init(void)
Packit Service d1fe03
{
Packit Service d1fe03
	xtables_register_matches(multiport_mt_reg, ARRAY_SIZE(multiport_mt_reg));
Packit Service d1fe03
}