xref: /aosp_15_r20/external/iptables/extensions/libarpt_mangle.c (revision a71a954618bbadd4a345637e5edcf36eec826889)
1*a71a9546SAutomerger Merge Worker /*
2*a71a9546SAutomerger Merge Worker  * Arturo Borrero Gonzalez <[email protected]> adapted
3*a71a9546SAutomerger Merge Worker  * this code to libxtables for arptables-compat in 2015
4*a71a9546SAutomerger Merge Worker  */
5*a71a9546SAutomerger Merge Worker 
6*a71a9546SAutomerger Merge Worker #include <stdio.h>
7*a71a9546SAutomerger Merge Worker #include <netdb.h>
8*a71a9546SAutomerger Merge Worker #include <string.h>
9*a71a9546SAutomerger Merge Worker #include <stdlib.h>
10*a71a9546SAutomerger Merge Worker #include <limits.h>
11*a71a9546SAutomerger Merge Worker #include <getopt.h>
12*a71a9546SAutomerger Merge Worker #include <netinet/ether.h>
13*a71a9546SAutomerger Merge Worker #include <xtables.h>
14*a71a9546SAutomerger Merge Worker #include <linux/netfilter_arp/arpt_mangle.h>
15*a71a9546SAutomerger Merge Worker #include "iptables/nft.h"
16*a71a9546SAutomerger Merge Worker 
arpmangle_print_help(void)17*a71a9546SAutomerger Merge Worker static void arpmangle_print_help(void)
18*a71a9546SAutomerger Merge Worker {
19*a71a9546SAutomerger Merge Worker 	printf(
20*a71a9546SAutomerger Merge Worker 	"mangle target options:\n"
21*a71a9546SAutomerger Merge Worker 	"--mangle-ip-s IP address\n"
22*a71a9546SAutomerger Merge Worker 	"--mangle-ip-d IP address\n"
23*a71a9546SAutomerger Merge Worker 	"--mangle-mac-s MAC address\n"
24*a71a9546SAutomerger Merge Worker 	"--mangle-mac-d MAC address\n"
25*a71a9546SAutomerger Merge Worker 	"--mangle-target target (DROP, CONTINUE or ACCEPT -- default is ACCEPT)\n");
26*a71a9546SAutomerger Merge Worker }
27*a71a9546SAutomerger Merge Worker 
28*a71a9546SAutomerger Merge Worker #define MANGLE_IPS    '1'
29*a71a9546SAutomerger Merge Worker #define MANGLE_IPT    '2'
30*a71a9546SAutomerger Merge Worker #define MANGLE_DEVS   '3'
31*a71a9546SAutomerger Merge Worker #define MANGLE_DEVT   '4'
32*a71a9546SAutomerger Merge Worker #define MANGLE_TARGET '5'
33*a71a9546SAutomerger Merge Worker 
34*a71a9546SAutomerger Merge Worker static const struct option arpmangle_opts[] = {
35*a71a9546SAutomerger Merge Worker 	{ .name = "mangle-ip-s",	.has_arg = true, .val = MANGLE_IPS },
36*a71a9546SAutomerger Merge Worker 	{ .name = "mangle-ip-d",	.has_arg = true, .val = MANGLE_IPT },
37*a71a9546SAutomerger Merge Worker 	{ .name = "mangle-mac-s",	.has_arg = true, .val = MANGLE_DEVS },
38*a71a9546SAutomerger Merge Worker 	{ .name = "mangle-mac-d",	.has_arg = true, .val = MANGLE_DEVT },
39*a71a9546SAutomerger Merge Worker 	{ .name = "mangle-target",	.has_arg = true, .val = MANGLE_TARGET },
40*a71a9546SAutomerger Merge Worker 	XT_GETOPT_TABLEEND,
41*a71a9546SAutomerger Merge Worker };
42*a71a9546SAutomerger Merge Worker 
arpmangle_init(struct xt_entry_target * target)43*a71a9546SAutomerger Merge Worker static void arpmangle_init(struct xt_entry_target *target)
44*a71a9546SAutomerger Merge Worker {
45*a71a9546SAutomerger Merge Worker 	struct arpt_mangle *mangle = (struct arpt_mangle *)target->data;
46*a71a9546SAutomerger Merge Worker 
47*a71a9546SAutomerger Merge Worker 	mangle->target = NF_ACCEPT;
48*a71a9546SAutomerger Merge Worker }
49*a71a9546SAutomerger Merge Worker 
50*a71a9546SAutomerger Merge Worker static int
arpmangle_parse(int c,char ** argv,int invert,unsigned int * flags,const void * entry,struct xt_entry_target ** target)51*a71a9546SAutomerger Merge Worker arpmangle_parse(int c, char **argv, int invert, unsigned int *flags,
52*a71a9546SAutomerger Merge Worker 		const void *entry, struct xt_entry_target **target)
53*a71a9546SAutomerger Merge Worker {
54*a71a9546SAutomerger Merge Worker 	struct arpt_mangle *mangle = (struct arpt_mangle *)(*target)->data;
55*a71a9546SAutomerger Merge Worker 	struct in_addr *ipaddr, mask;
56*a71a9546SAutomerger Merge Worker 	struct ether_addr *macaddr;
57*a71a9546SAutomerger Merge Worker 	const struct arpt_entry *e = (const struct arpt_entry *)entry;
58*a71a9546SAutomerger Merge Worker 	unsigned int nr;
59*a71a9546SAutomerger Merge Worker 	int ret = 1;
60*a71a9546SAutomerger Merge Worker 
61*a71a9546SAutomerger Merge Worker 	memset(&mask, 0, sizeof(mask));
62*a71a9546SAutomerger Merge Worker 
63*a71a9546SAutomerger Merge Worker 	switch (c) {
64*a71a9546SAutomerger Merge Worker 	case MANGLE_IPS:
65*a71a9546SAutomerger Merge Worker 		xtables_ipparse_any(optarg, &ipaddr, &mask, &nr);
66*a71a9546SAutomerger Merge Worker 		mangle->u_s.src_ip.s_addr = ipaddr->s_addr;
67*a71a9546SAutomerger Merge Worker 		free(ipaddr);
68*a71a9546SAutomerger Merge Worker 		mangle->flags |= ARPT_MANGLE_SIP;
69*a71a9546SAutomerger Merge Worker 		break;
70*a71a9546SAutomerger Merge Worker 	case MANGLE_IPT:
71*a71a9546SAutomerger Merge Worker 		xtables_ipparse_any(optarg, &ipaddr, &mask, &nr);
72*a71a9546SAutomerger Merge Worker 		mangle->u_t.tgt_ip.s_addr = ipaddr->s_addr;
73*a71a9546SAutomerger Merge Worker 		free(ipaddr);
74*a71a9546SAutomerger Merge Worker 		mangle->flags |= ARPT_MANGLE_TIP;
75*a71a9546SAutomerger Merge Worker 		break;
76*a71a9546SAutomerger Merge Worker 	case MANGLE_DEVS:
77*a71a9546SAutomerger Merge Worker 		if (e->arp.arhln_mask == 0)
78*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
79*a71a9546SAutomerger Merge Worker 				      "no --h-length defined");
80*a71a9546SAutomerger Merge Worker 		if (e->arp.invflags & ARPT_INV_ARPHLN)
81*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
82*a71a9546SAutomerger Merge Worker 				      "! --h-length not allowed for "
83*a71a9546SAutomerger Merge Worker 				      "--mangle-mac-s");
84*a71a9546SAutomerger Merge Worker 		if (e->arp.arhln != 6)
85*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
86*a71a9546SAutomerger Merge Worker 				      "only --h-length 6 supported");
87*a71a9546SAutomerger Merge Worker 		macaddr = ether_aton(optarg);
88*a71a9546SAutomerger Merge Worker 		if (macaddr == NULL)
89*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
90*a71a9546SAutomerger Merge Worker 				      "invalid source MAC");
91*a71a9546SAutomerger Merge Worker 		memcpy(mangle->src_devaddr, macaddr, e->arp.arhln);
92*a71a9546SAutomerger Merge Worker 		mangle->flags |= ARPT_MANGLE_SDEV;
93*a71a9546SAutomerger Merge Worker 		break;
94*a71a9546SAutomerger Merge Worker 	case MANGLE_DEVT:
95*a71a9546SAutomerger Merge Worker 		if (e->arp.arhln_mask == 0)
96*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
97*a71a9546SAutomerger Merge Worker 				      "no --h-length defined");
98*a71a9546SAutomerger Merge Worker 		if (e->arp.invflags & ARPT_INV_ARPHLN)
99*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
100*a71a9546SAutomerger Merge Worker 				      "! hln not allowed for --mangle-mac-d");
101*a71a9546SAutomerger Merge Worker 		if (e->arp.arhln != 6)
102*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
103*a71a9546SAutomerger Merge Worker 				      "only --h-length 6 supported");
104*a71a9546SAutomerger Merge Worker 		macaddr = ether_aton(optarg);
105*a71a9546SAutomerger Merge Worker 		if (macaddr == NULL)
106*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM, "invalid target MAC");
107*a71a9546SAutomerger Merge Worker 		memcpy(mangle->tgt_devaddr, macaddr, e->arp.arhln);
108*a71a9546SAutomerger Merge Worker 		mangle->flags |= ARPT_MANGLE_TDEV;
109*a71a9546SAutomerger Merge Worker 		break;
110*a71a9546SAutomerger Merge Worker 	case MANGLE_TARGET:
111*a71a9546SAutomerger Merge Worker 		if (!strcmp(optarg, "DROP"))
112*a71a9546SAutomerger Merge Worker 			mangle->target = NF_DROP;
113*a71a9546SAutomerger Merge Worker 		else if (!strcmp(optarg, "ACCEPT"))
114*a71a9546SAutomerger Merge Worker 			mangle->target = NF_ACCEPT;
115*a71a9546SAutomerger Merge Worker 		else if (!strcmp(optarg, "CONTINUE"))
116*a71a9546SAutomerger Merge Worker 			mangle->target = XT_CONTINUE;
117*a71a9546SAutomerger Merge Worker 		else
118*a71a9546SAutomerger Merge Worker 			xtables_error(PARAMETER_PROBLEM,
119*a71a9546SAutomerger Merge Worker 				      "bad target for --mangle-target");
120*a71a9546SAutomerger Merge Worker 		break;
121*a71a9546SAutomerger Merge Worker 	default:
122*a71a9546SAutomerger Merge Worker 		ret = 0;
123*a71a9546SAutomerger Merge Worker 	}
124*a71a9546SAutomerger Merge Worker 
125*a71a9546SAutomerger Merge Worker 	return ret;
126*a71a9546SAutomerger Merge Worker }
127*a71a9546SAutomerger Merge Worker 
arpmangle_final_check(unsigned int flags)128*a71a9546SAutomerger Merge Worker static void arpmangle_final_check(unsigned int flags)
129*a71a9546SAutomerger Merge Worker {
130*a71a9546SAutomerger Merge Worker }
131*a71a9546SAutomerger Merge Worker 
ipaddr_to(const struct in_addr * addrp,int numeric)132*a71a9546SAutomerger Merge Worker static const char *ipaddr_to(const struct in_addr *addrp, int numeric)
133*a71a9546SAutomerger Merge Worker {
134*a71a9546SAutomerger Merge Worker 	if (numeric)
135*a71a9546SAutomerger Merge Worker 		return xtables_ipaddr_to_numeric(addrp);
136*a71a9546SAutomerger Merge Worker 	else
137*a71a9546SAutomerger Merge Worker 		return xtables_ipaddr_to_anyname(addrp);
138*a71a9546SAutomerger Merge Worker }
139*a71a9546SAutomerger Merge Worker 
140*a71a9546SAutomerger Merge Worker static void
arpmangle_print(const void * ip,const struct xt_entry_target * target,int numeric)141*a71a9546SAutomerger Merge Worker arpmangle_print(const void *ip, const struct xt_entry_target *target,
142*a71a9546SAutomerger Merge Worker 		int numeric)
143*a71a9546SAutomerger Merge Worker {
144*a71a9546SAutomerger Merge Worker 	struct arpt_mangle *m = (struct arpt_mangle *)(target->data);
145*a71a9546SAutomerger Merge Worker 
146*a71a9546SAutomerger Merge Worker 	if (m->flags & ARPT_MANGLE_SIP) {
147*a71a9546SAutomerger Merge Worker 		printf(" --mangle-ip-s %s",
148*a71a9546SAutomerger Merge Worker 		       ipaddr_to(&(m->u_s.src_ip), numeric));
149*a71a9546SAutomerger Merge Worker 	}
150*a71a9546SAutomerger Merge Worker 	if (m->flags & ARPT_MANGLE_SDEV) {
151*a71a9546SAutomerger Merge Worker 		printf(" --mangle-mac-s ");
152*a71a9546SAutomerger Merge Worker 		xtables_print_mac((unsigned char *)m->src_devaddr);
153*a71a9546SAutomerger Merge Worker 	}
154*a71a9546SAutomerger Merge Worker 	if (m->flags & ARPT_MANGLE_TIP) {
155*a71a9546SAutomerger Merge Worker 		printf(" --mangle-ip-d %s",
156*a71a9546SAutomerger Merge Worker 		       ipaddr_to(&(m->u_t.tgt_ip), numeric));
157*a71a9546SAutomerger Merge Worker 	}
158*a71a9546SAutomerger Merge Worker 	if (m->flags & ARPT_MANGLE_TDEV) {
159*a71a9546SAutomerger Merge Worker 		printf(" --mangle-mac-d ");
160*a71a9546SAutomerger Merge Worker 		xtables_print_mac((unsigned char *)m->tgt_devaddr);
161*a71a9546SAutomerger Merge Worker 	}
162*a71a9546SAutomerger Merge Worker 	if (m->target != NF_ACCEPT) {
163*a71a9546SAutomerger Merge Worker 		printf(" --mangle-target %s",
164*a71a9546SAutomerger Merge Worker 		       m->target == NF_DROP ? "DROP" : "CONTINUE");
165*a71a9546SAutomerger Merge Worker 	}
166*a71a9546SAutomerger Merge Worker }
167*a71a9546SAutomerger Merge Worker 
arpmangle_save(const void * ip,const struct xt_entry_target * target)168*a71a9546SAutomerger Merge Worker static void arpmangle_save(const void *ip, const struct xt_entry_target *target)
169*a71a9546SAutomerger Merge Worker {
170*a71a9546SAutomerger Merge Worker 	arpmangle_print(ip, target, 0);
171*a71a9546SAutomerger Merge Worker }
172*a71a9546SAutomerger Merge Worker 
173*a71a9546SAutomerger Merge Worker static struct xtables_target arpmangle_target = {
174*a71a9546SAutomerger Merge Worker 	.name		= "mangle",
175*a71a9546SAutomerger Merge Worker 	.revision	= 0,
176*a71a9546SAutomerger Merge Worker 	.version	= XTABLES_VERSION,
177*a71a9546SAutomerger Merge Worker 	.family		= NFPROTO_ARP,
178*a71a9546SAutomerger Merge Worker 	.size		= XT_ALIGN(sizeof(struct arpt_mangle)),
179*a71a9546SAutomerger Merge Worker 	.userspacesize	= XT_ALIGN(sizeof(struct arpt_mangle)),
180*a71a9546SAutomerger Merge Worker 	.help		= arpmangle_print_help,
181*a71a9546SAutomerger Merge Worker 	.init		= arpmangle_init,
182*a71a9546SAutomerger Merge Worker 	.parse		= arpmangle_parse,
183*a71a9546SAutomerger Merge Worker 	.final_check	= arpmangle_final_check,
184*a71a9546SAutomerger Merge Worker 	.print		= arpmangle_print,
185*a71a9546SAutomerger Merge Worker 	.save		= arpmangle_save,
186*a71a9546SAutomerger Merge Worker 	.extra_opts	= arpmangle_opts,
187*a71a9546SAutomerger Merge Worker };
188*a71a9546SAutomerger Merge Worker 
_init(void)189*a71a9546SAutomerger Merge Worker void _init(void)
190*a71a9546SAutomerger Merge Worker {
191*a71a9546SAutomerger Merge Worker 	xtables_register_target(&arpmangle_target);
192*a71a9546SAutomerger Merge Worker }
193