xref: /linux/net/ipv4/netfilter/arpt_mangle.c (revision 3862c6a9)
109c434b8SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
21da177e4SLinus Torvalds /* module that allows mangling of the arp payload */
31da177e4SLinus Torvalds #include <linux/module.h>
42ca7b0acSHerbert Xu #include <linux/netfilter.h>
51da177e4SLinus Torvalds #include <linux/netfilter_arp/arpt_mangle.h>
61da177e4SLinus Torvalds #include <net/sock.h>
71da177e4SLinus Torvalds 
81da177e4SLinus Torvalds MODULE_LICENSE("GPL");
91da177e4SLinus Torvalds MODULE_AUTHOR("Bart De Schuymer <bdschuym@pandora.be>");
101da177e4SLinus Torvalds MODULE_DESCRIPTION("arptables arp payload mangle target");
111da177e4SLinus Torvalds 
121da177e4SLinus Torvalds static unsigned int
target(struct sk_buff * skb,const struct xt_action_param * par)134b560b44SJan Engelhardt target(struct sk_buff *skb, const struct xt_action_param *par)
141da177e4SLinus Torvalds {
157eb35586SJan Engelhardt 	const struct arpt_mangle *mangle = par->targinfo;
165452e425SJan Engelhardt 	const struct arphdr *arp;
171da177e4SLinus Torvalds 	unsigned char *arpptr;
181da177e4SLinus Torvalds 	int pln, hln;
191da177e4SLinus Torvalds 
20*3862c6a9SFlorian Westphal 	if (skb_ensure_writable(skb, skb->len))
211da177e4SLinus Torvalds 		return NF_DROP;
221da177e4SLinus Torvalds 
233db05feaSHerbert Xu 	arp = arp_hdr(skb);
243db05feaSHerbert Xu 	arpptr = skb_network_header(skb) + sizeof(*arp);
251da177e4SLinus Torvalds 	pln = arp->ar_pln;
261da177e4SLinus Torvalds 	hln = arp->ar_hln;
271da177e4SLinus Torvalds 	/* We assume that pln and hln were checked in the match */
281da177e4SLinus Torvalds 	if (mangle->flags & ARPT_MANGLE_SDEV) {
291da177e4SLinus Torvalds 		if (ARPT_DEV_ADDR_LEN_MAX < hln ||
303db05feaSHerbert Xu 		   (arpptr + hln > skb_tail_pointer(skb)))
311da177e4SLinus Torvalds 			return NF_DROP;
321da177e4SLinus Torvalds 		memcpy(arpptr, mangle->src_devaddr, hln);
331da177e4SLinus Torvalds 	}
341da177e4SLinus Torvalds 	arpptr += hln;
351da177e4SLinus Torvalds 	if (mangle->flags & ARPT_MANGLE_SIP) {
361da177e4SLinus Torvalds 		if (ARPT_MANGLE_ADDR_LEN_MAX < pln ||
373db05feaSHerbert Xu 		   (arpptr + pln > skb_tail_pointer(skb)))
381da177e4SLinus Torvalds 			return NF_DROP;
391da177e4SLinus Torvalds 		memcpy(arpptr, &mangle->u_s.src_ip, pln);
401da177e4SLinus Torvalds 	}
411da177e4SLinus Torvalds 	arpptr += pln;
421da177e4SLinus Torvalds 	if (mangle->flags & ARPT_MANGLE_TDEV) {
431da177e4SLinus Torvalds 		if (ARPT_DEV_ADDR_LEN_MAX < hln ||
443db05feaSHerbert Xu 		   (arpptr + hln > skb_tail_pointer(skb)))
451da177e4SLinus Torvalds 			return NF_DROP;
461da177e4SLinus Torvalds 		memcpy(arpptr, mangle->tgt_devaddr, hln);
471da177e4SLinus Torvalds 	}
481da177e4SLinus Torvalds 	arpptr += hln;
491da177e4SLinus Torvalds 	if (mangle->flags & ARPT_MANGLE_TIP) {
501da177e4SLinus Torvalds 		if (ARPT_MANGLE_ADDR_LEN_MAX < pln ||
513db05feaSHerbert Xu 		   (arpptr + pln > skb_tail_pointer(skb)))
521da177e4SLinus Torvalds 			return NF_DROP;
531da177e4SLinus Torvalds 		memcpy(arpptr, &mangle->u_t.tgt_ip, pln);
541da177e4SLinus Torvalds 	}
551da177e4SLinus Torvalds 	return mangle->target;
561da177e4SLinus Torvalds }
571da177e4SLinus Torvalds 
checkentry(const struct xt_tgchk_param * par)58135367b8SJan Engelhardt static int checkentry(const struct xt_tgchk_param *par)
591da177e4SLinus Torvalds {
60af5d6dc2SJan Engelhardt 	const struct arpt_mangle *mangle = par->targinfo;
611da177e4SLinus Torvalds 
621da177e4SLinus Torvalds 	if (mangle->flags & ~ARPT_MANGLE_MASK ||
631da177e4SLinus Torvalds 	    !(mangle->flags & ARPT_MANGLE_MASK))
649d0db8b6SPablo Neira Ayuso 		return -EINVAL;
651da177e4SLinus Torvalds 
661da177e4SLinus Torvalds 	if (mangle->target != NF_DROP && mangle->target != NF_ACCEPT &&
67243bf6e2SJan Engelhardt 	   mangle->target != XT_CONTINUE)
689d0db8b6SPablo Neira Ayuso 		return -EINVAL;
699d0db8b6SPablo Neira Ayuso 	return 0;
701da177e4SLinus Torvalds }
711da177e4SLinus Torvalds 
7295eea855SJan Engelhardt static struct xt_target arpt_mangle_reg __read_mostly = {
731da177e4SLinus Torvalds 	.name		= "mangle",
74ee999d8bSJan Engelhardt 	.family		= NFPROTO_ARP,
751da177e4SLinus Torvalds 	.target		= target,
76aa83c1abSPatrick McHardy 	.targetsize	= sizeof(struct arpt_mangle),
771da177e4SLinus Torvalds 	.checkentry	= checkentry,
781da177e4SLinus Torvalds 	.me		= THIS_MODULE,
791da177e4SLinus Torvalds };
801da177e4SLinus Torvalds 
arpt_mangle_init(void)8165b4b4e8SAndrew Morton static int __init arpt_mangle_init(void)
821da177e4SLinus Torvalds {
833bb0362dSJan Engelhardt 	return xt_register_target(&arpt_mangle_reg);
841da177e4SLinus Torvalds }
851da177e4SLinus Torvalds 
arpt_mangle_fini(void)8665b4b4e8SAndrew Morton static void __exit arpt_mangle_fini(void)
871da177e4SLinus Torvalds {
883bb0362dSJan Engelhardt 	xt_unregister_target(&arpt_mangle_reg);
891da177e4SLinus Torvalds }
901da177e4SLinus Torvalds 
9165b4b4e8SAndrew Morton module_init(arpt_mangle_init);
9265b4b4e8SAndrew Morton module_exit(arpt_mangle_fini);
93