Blame extensions/libarpt_mangle.c

Packit 7b22a4
/*
Packit 7b22a4
 * Arturo Borrero Gonzalez <arturo@debian.org> adapted
Packit 7b22a4
 * this code to libxtables for arptables-compat in 2015
Packit 7b22a4
 */
Packit 7b22a4
Packit 7b22a4
#include <stdio.h>
Packit 7b22a4
#include <netdb.h>
Packit 7b22a4
#include <string.h>
Packit 7b22a4
#include <stdlib.h>
Packit 7b22a4
#include <limits.h>
Packit 7b22a4
#include <getopt.h>
Packit 7b22a4
#include <netinet/ether.h>
Packit 7b22a4
#include <xtables.h>
Packit 7b22a4
#include <linux/netfilter_arp/arpt_mangle.h>
Packit 7b22a4
#include "iptables/nft.h"
Packit 7b22a4
#include "iptables/nft-arp.h"
Packit 7b22a4
Packit 7b22a4
static void arpmangle_print_help(void)
Packit 7b22a4
{
Packit 7b22a4
	printf(
Packit 7b22a4
	"mangle target options:\n"
Packit 7b22a4
	"--mangle-ip-s IP address\n"
Packit 7b22a4
	"--mangle-ip-d IP address\n"
Packit 7b22a4
	"--mangle-mac-s MAC address\n"
Packit 7b22a4
	"--mangle-mac-d MAC address\n"
Packit 7b22a4
	"--mangle-target target (DROP, CONTINUE or ACCEPT -- default is ACCEPT)\n");
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
#define MANGLE_IPS    '1'
Packit 7b22a4
#define MANGLE_IPT    '2'
Packit 7b22a4
#define MANGLE_DEVS   '3'
Packit 7b22a4
#define MANGLE_DEVT   '4'
Packit 7b22a4
#define MANGLE_TARGET '5'
Packit 7b22a4
Packit 7b22a4
static const struct option arpmangle_opts[] = {
Packit 7b22a4
	{ .name = "mangle-ip-s",	.has_arg = true, .val = MANGLE_IPS },
Packit 7b22a4
	{ .name = "mangle-ip-d",	.has_arg = true, .val = MANGLE_IPT },
Packit 7b22a4
	{ .name = "mangle-mac-s",	.has_arg = true, .val = MANGLE_DEVS },
Packit 7b22a4
	{ .name = "mangle-mac-d",	.has_arg = true, .val = MANGLE_DEVT },
Packit 7b22a4
	{ .name = "mangle-target",	.has_arg = true, .val = MANGLE_TARGET },
Packit 7b22a4
	XT_GETOPT_TABLEEND,
Packit 7b22a4
};
Packit 7b22a4
Packit 7b22a4
static void arpmangle_init(struct xt_entry_target *target)
Packit 7b22a4
{
Packit 7b22a4
	struct arpt_mangle *mangle = (struct arpt_mangle *)target->data;
Packit 7b22a4
Packit 7b22a4
	mangle->target = NF_ACCEPT;
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
static int
Packit 7b22a4
arpmangle_parse(int c, char **argv, int invert, unsigned int *flags,
Packit 7b22a4
		const void *entry, struct xt_entry_target **target)
Packit 7b22a4
{
Packit 7b22a4
	struct arpt_mangle *mangle = (struct arpt_mangle *)(*target)->data;
Packit 7b22a4
	struct in_addr *ipaddr, mask;
Packit 7b22a4
	struct ether_addr *macaddr;
Packit 7b22a4
	const struct arpt_entry *e = (const struct arpt_entry *)entry;
Packit 7b22a4
	unsigned int nr;
Packit 7b22a4
	int ret = 1;
Packit 7b22a4
Packit 7b22a4
	memset(&mask, 0, sizeof(mask));
Packit 7b22a4
Packit 7b22a4
	switch (c) {
Packit 7b22a4
	case MANGLE_IPS:
Packit 7b22a4
		xtables_ipparse_any(optarg, &ipaddr, &mask, &nr);
Packit 7b22a4
		mangle->u_s.src_ip.s_addr = ipaddr->s_addr;
Packit 7b22a4
		free(ipaddr);
Packit 7b22a4
		mangle->flags |= ARPT_MANGLE_SIP;
Packit 7b22a4
		break;
Packit 7b22a4
	case MANGLE_IPT:
Packit 7b22a4
		xtables_ipparse_any(optarg, &ipaddr, &mask, &nr);
Packit 7b22a4
		mangle->u_t.tgt_ip.s_addr = ipaddr->s_addr;
Packit 7b22a4
		free(ipaddr);
Packit 7b22a4
		mangle->flags |= ARPT_MANGLE_TIP;
Packit 7b22a4
		break;
Packit 7b22a4
	case MANGLE_DEVS:
Packit 7b22a4
		if (e->arp.arhln_mask == 0)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "no --h-length defined");
Packit 7b22a4
		if (e->arp.invflags & ARPT_INV_ARPHLN)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "! --h-length not allowed for "
Packit 7b22a4
				      "--mangle-mac-s");
Packit 7b22a4
		if (e->arp.arhln != 6)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "only --h-length 6 supported");
Packit 7b22a4
		macaddr = ether_aton(optarg);
Packit 7b22a4
		if (macaddr == NULL)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "invalid source MAC");
Packit 7b22a4
		memcpy(mangle->src_devaddr, macaddr, e->arp.arhln);
Packit 7b22a4
		mangle->flags |= ARPT_MANGLE_SDEV;
Packit 7b22a4
		break;
Packit 7b22a4
	case MANGLE_DEVT:
Packit 7b22a4
		if (e->arp.arhln_mask == 0)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "no --h-length defined");
Packit 7b22a4
		if (e->arp.invflags & ARPT_INV_ARPHLN)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "! hln not allowed for --mangle-mac-d");
Packit 7b22a4
		if (e->arp.arhln != 6)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "only --h-length 6 supported");
Packit 7b22a4
		macaddr = ether_aton(optarg);
Packit 7b22a4
		if (macaddr == NULL)
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM, "invalid target MAC");
Packit 7b22a4
		memcpy(mangle->tgt_devaddr, macaddr, e->arp.arhln);
Packit 7b22a4
		mangle->flags |= ARPT_MANGLE_TDEV;
Packit 7b22a4
		break;
Packit 7b22a4
	case MANGLE_TARGET:
Packit 7b22a4
		if (!strcmp(optarg, "DROP"))
Packit 7b22a4
			mangle->target = NF_DROP;
Packit 7b22a4
		else if (!strcmp(optarg, "ACCEPT"))
Packit 7b22a4
			mangle->target = NF_ACCEPT;
Packit 7b22a4
		else if (!strcmp(optarg, "CONTINUE"))
Packit 7b22a4
			mangle->target = XT_CONTINUE;
Packit 7b22a4
		else
Packit 7b22a4
			xtables_error(PARAMETER_PROBLEM,
Packit 7b22a4
				      "bad target for --mangle-target");
Packit 7b22a4
		break;
Packit 7b22a4
	default:
Packit 7b22a4
		ret = 0;
Packit 7b22a4
	}
Packit 7b22a4
Packit 7b22a4
	return ret;
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
static void arpmangle_final_check(unsigned int flags)
Packit 7b22a4
{
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
static void print_mac(const unsigned char *mac, int l)
Packit 7b22a4
{
Packit 7b22a4
	int j;
Packit 7b22a4
Packit 7b22a4
	for (j = 0; j < l; j++)
Packit 7b22a4
		printf("%02x%s", mac[j],
Packit 7b22a4
			(j==l-1) ? "" : ":");
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
static const char *ipaddr_to(const struct in_addr *addrp, int numeric)
Packit 7b22a4
{
Packit 7b22a4
	if (numeric)
Packit 7b22a4
		return xtables_ipaddr_to_numeric(addrp);
Packit 7b22a4
	else
Packit 7b22a4
		return xtables_ipaddr_to_anyname(addrp);
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
static void
Packit 7b22a4
arpmangle_print(const void *ip, const struct xt_entry_target *target,
Packit 7b22a4
		int numeric)
Packit 7b22a4
{
Packit 7b22a4
	struct arpt_mangle *m = (struct arpt_mangle *)(target->data);
Packit 7b22a4
Packit 7b22a4
	if (m->flags & ARPT_MANGLE_SIP) {
Packit 7b22a4
		printf(" --mangle-ip-s %s",
Packit 7b22a4
		       ipaddr_to(&(m->u_s.src_ip), numeric));
Packit 7b22a4
	}
Packit 7b22a4
	if (m->flags & ARPT_MANGLE_SDEV) {
Packit 7b22a4
		printf(" --mangle-mac-s ");
Packit 7b22a4
		print_mac((unsigned char *)m->src_devaddr, 6);
Packit 7b22a4
	}
Packit 7b22a4
	if (m->flags & ARPT_MANGLE_TIP) {
Packit 7b22a4
		printf(" --mangle-ip-d %s",
Packit 7b22a4
		       ipaddr_to(&(m->u_t.tgt_ip), numeric));
Packit 7b22a4
	}
Packit 7b22a4
	if (m->flags & ARPT_MANGLE_TDEV) {
Packit 7b22a4
		printf(" --mangle-mac-d ");
Packit 7b22a4
		print_mac((unsigned char *)m->tgt_devaddr, 6);
Packit 7b22a4
	}
Packit 7b22a4
	if (m->target != NF_ACCEPT) {
Packit 7b22a4
		printf(" --mangle-target %s",
Packit 7b22a4
		       m->target == NF_DROP ? "DROP" : "CONTINUE");
Packit 7b22a4
	}
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
static void arpmangle_save(const void *ip, const struct xt_entry_target *target)
Packit 7b22a4
{
Packit 7b22a4
	arpmangle_print(ip, target, 0);
Packit 7b22a4
}
Packit 7b22a4
Packit 7b22a4
static struct xtables_target arpmangle_target = {
Packit 7b22a4
	.name		= "mangle",
Packit 7b22a4
	.revision	= 0,
Packit 7b22a4
	.version	= XTABLES_VERSION,
Packit 7b22a4
	.family		= NFPROTO_ARP,
Packit 7b22a4
	.size		= XT_ALIGN(sizeof(struct arpt_mangle)),
Packit 7b22a4
	.userspacesize	= XT_ALIGN(sizeof(struct arpt_mangle)),
Packit 7b22a4
	.help		= arpmangle_print_help,
Packit 7b22a4
	.init		= arpmangle_init,
Packit 7b22a4
	.parse		= arpmangle_parse,
Packit 7b22a4
	.final_check	= arpmangle_final_check,
Packit 7b22a4
	.print		= arpmangle_print,
Packit 7b22a4
	.save		= arpmangle_save,
Packit 7b22a4
	.extra_opts	= arpmangle_opts,
Packit 7b22a4
};
Packit 7b22a4
Packit 7b22a4
void _init(void)
Packit 7b22a4
{
Packit 7b22a4
	xtables_register_target(&arpmangle_target);
Packit 7b22a4
}