xref: /aosp_15_r20/external/iptables/extensions/libxt_TEE.c (revision a71a954618bbadd4a345637e5edcf36eec826889)
1*a71a9546SAutomerger Merge Worker /*
2*a71a9546SAutomerger Merge Worker  *	"TEE" target extension for iptables
3*a71a9546SAutomerger Merge Worker  *	Copyright © Sebastian Claßen <sebastian.classen [at] freenet.ag>, 2007
4*a71a9546SAutomerger Merge Worker  *	Jan Engelhardt <jengelh [at] medozas de>, 2007 - 2010
5*a71a9546SAutomerger Merge Worker  *
6*a71a9546SAutomerger Merge Worker  *	This program is free software; you can redistribute it and/or
7*a71a9546SAutomerger Merge Worker  *	modify it under the terms of the GNU General Public License; either
8*a71a9546SAutomerger Merge Worker  *	version 2 of the License, or any later version, as published by the
9*a71a9546SAutomerger Merge Worker  *	Free Software Foundation.
10*a71a9546SAutomerger Merge Worker  */
11*a71a9546SAutomerger Merge Worker #include <sys/socket.h>
12*a71a9546SAutomerger Merge Worker #include <getopt.h>
13*a71a9546SAutomerger Merge Worker #include <stdbool.h>
14*a71a9546SAutomerger Merge Worker #include <stdio.h>
15*a71a9546SAutomerger Merge Worker #include <stdlib.h>
16*a71a9546SAutomerger Merge Worker #include <string.h>
17*a71a9546SAutomerger Merge Worker 
18*a71a9546SAutomerger Merge Worker #include <arpa/inet.h>
19*a71a9546SAutomerger Merge Worker #include <net/if.h>
20*a71a9546SAutomerger Merge Worker #include <netinet/in.h>
21*a71a9546SAutomerger Merge Worker 
22*a71a9546SAutomerger Merge Worker #include <xtables.h>
23*a71a9546SAutomerger Merge Worker #include <linux/netfilter.h>
24*a71a9546SAutomerger Merge Worker #include <linux/netfilter/x_tables.h>
25*a71a9546SAutomerger Merge Worker #include <linux/netfilter/xt_TEE.h>
26*a71a9546SAutomerger Merge Worker 
27*a71a9546SAutomerger Merge Worker enum {
28*a71a9546SAutomerger Merge Worker 	O_GATEWAY = 0,
29*a71a9546SAutomerger Merge Worker 	O_OIF,
30*a71a9546SAutomerger Merge Worker };
31*a71a9546SAutomerger Merge Worker 
32*a71a9546SAutomerger Merge Worker #define s struct xt_tee_tginfo
33*a71a9546SAutomerger Merge Worker static const struct xt_option_entry tee_tg_opts[] = {
34*a71a9546SAutomerger Merge Worker 	{.name = "gateway", .id = O_GATEWAY, .type = XTTYPE_HOST,
35*a71a9546SAutomerger Merge Worker 	 .flags = XTOPT_MAND | XTOPT_PUT, XTOPT_POINTER(s, gw)},
36*a71a9546SAutomerger Merge Worker 	{.name = "oif", .id = O_OIF, .type = XTTYPE_STRING,
37*a71a9546SAutomerger Merge Worker 	 .flags = XTOPT_PUT, XTOPT_POINTER(s, oif)},
38*a71a9546SAutomerger Merge Worker 	XTOPT_TABLEEND,
39*a71a9546SAutomerger Merge Worker };
40*a71a9546SAutomerger Merge Worker #undef s
41*a71a9546SAutomerger Merge Worker 
tee_tg_help(void)42*a71a9546SAutomerger Merge Worker static void tee_tg_help(void)
43*a71a9546SAutomerger Merge Worker {
44*a71a9546SAutomerger Merge Worker 	printf(
45*a71a9546SAutomerger Merge Worker "TEE target options:\n"
46*a71a9546SAutomerger Merge Worker "  --gateway IPADDR    Route packet via the gateway given by address\n"
47*a71a9546SAutomerger Merge Worker "  --oif NAME          Include oif in route calculation\n"
48*a71a9546SAutomerger Merge Worker "\n");
49*a71a9546SAutomerger Merge Worker }
50*a71a9546SAutomerger Merge Worker 
tee_tg_print(const void * ip,const struct xt_entry_target * target,int numeric)51*a71a9546SAutomerger Merge Worker static void tee_tg_print(const void *ip, const struct xt_entry_target *target,
52*a71a9546SAutomerger Merge Worker                          int numeric)
53*a71a9546SAutomerger Merge Worker {
54*a71a9546SAutomerger Merge Worker 	const struct xt_tee_tginfo *info = (const void *)target->data;
55*a71a9546SAutomerger Merge Worker 
56*a71a9546SAutomerger Merge Worker 	if (numeric)
57*a71a9546SAutomerger Merge Worker 		printf(" TEE gw:%s", xtables_ipaddr_to_numeric(&info->gw.in));
58*a71a9546SAutomerger Merge Worker 	else
59*a71a9546SAutomerger Merge Worker 		printf(" TEE gw:%s", xtables_ipaddr_to_anyname(&info->gw.in));
60*a71a9546SAutomerger Merge Worker 	if (*info->oif != '\0')
61*a71a9546SAutomerger Merge Worker 		printf(" oif=%s", info->oif);
62*a71a9546SAutomerger Merge Worker }
63*a71a9546SAutomerger Merge Worker 
tee_tg6_print(const void * ip,const struct xt_entry_target * target,int numeric)64*a71a9546SAutomerger Merge Worker static void tee_tg6_print(const void *ip, const struct xt_entry_target *target,
65*a71a9546SAutomerger Merge Worker                           int numeric)
66*a71a9546SAutomerger Merge Worker {
67*a71a9546SAutomerger Merge Worker 	const struct xt_tee_tginfo *info = (const void *)target->data;
68*a71a9546SAutomerger Merge Worker 
69*a71a9546SAutomerger Merge Worker 	if (numeric)
70*a71a9546SAutomerger Merge Worker 		printf(" TEE gw:%s", xtables_ip6addr_to_numeric(&info->gw.in6));
71*a71a9546SAutomerger Merge Worker 	else
72*a71a9546SAutomerger Merge Worker 		printf(" TEE gw:%s", xtables_ip6addr_to_anyname(&info->gw.in6));
73*a71a9546SAutomerger Merge Worker 	if (*info->oif != '\0')
74*a71a9546SAutomerger Merge Worker 		printf(" oif=%s", info->oif);
75*a71a9546SAutomerger Merge Worker }
76*a71a9546SAutomerger Merge Worker 
tee_tg_save(const void * ip,const struct xt_entry_target * target)77*a71a9546SAutomerger Merge Worker static void tee_tg_save(const void *ip, const struct xt_entry_target *target)
78*a71a9546SAutomerger Merge Worker {
79*a71a9546SAutomerger Merge Worker 	const struct xt_tee_tginfo *info = (const void *)target->data;
80*a71a9546SAutomerger Merge Worker 
81*a71a9546SAutomerger Merge Worker 	printf(" --gateway %s", xtables_ipaddr_to_numeric(&info->gw.in));
82*a71a9546SAutomerger Merge Worker 	if (*info->oif != '\0')
83*a71a9546SAutomerger Merge Worker 		printf(" --oif %s", info->oif);
84*a71a9546SAutomerger Merge Worker }
85*a71a9546SAutomerger Merge Worker 
tee_tg6_save(const void * ip,const struct xt_entry_target * target)86*a71a9546SAutomerger Merge Worker static void tee_tg6_save(const void *ip, const struct xt_entry_target *target)
87*a71a9546SAutomerger Merge Worker {
88*a71a9546SAutomerger Merge Worker 	const struct xt_tee_tginfo *info = (const void *)target->data;
89*a71a9546SAutomerger Merge Worker 
90*a71a9546SAutomerger Merge Worker 	printf(" --gateway %s", xtables_ip6addr_to_numeric(&info->gw.in6));
91*a71a9546SAutomerger Merge Worker 	if (*info->oif != '\0')
92*a71a9546SAutomerger Merge Worker 		printf(" --oif %s", info->oif);
93*a71a9546SAutomerger Merge Worker }
94*a71a9546SAutomerger Merge Worker 
tee_tg_xlate(struct xt_xlate * xl,const struct xt_xlate_tg_params * params)95*a71a9546SAutomerger Merge Worker static int tee_tg_xlate(struct xt_xlate *xl,
96*a71a9546SAutomerger Merge Worker 			const struct xt_xlate_tg_params *params)
97*a71a9546SAutomerger Merge Worker {
98*a71a9546SAutomerger Merge Worker 	const struct xt_tee_tginfo *info = (const void *)params->target->data;
99*a71a9546SAutomerger Merge Worker 
100*a71a9546SAutomerger Merge Worker 	if (params->numeric)
101*a71a9546SAutomerger Merge Worker 		xt_xlate_add(xl, "dup to %s",
102*a71a9546SAutomerger Merge Worker 			     xtables_ipaddr_to_numeric(&info->gw.in));
103*a71a9546SAutomerger Merge Worker 	else
104*a71a9546SAutomerger Merge Worker 		xt_xlate_add(xl, "dup to %s",
105*a71a9546SAutomerger Merge Worker 			     xtables_ipaddr_to_anyname(&info->gw.in));
106*a71a9546SAutomerger Merge Worker 	if (*info->oif != '\0')
107*a71a9546SAutomerger Merge Worker 		xt_xlate_add(xl, " device %s", info->oif);
108*a71a9546SAutomerger Merge Worker 
109*a71a9546SAutomerger Merge Worker 	return 1;
110*a71a9546SAutomerger Merge Worker }
111*a71a9546SAutomerger Merge Worker 
tee_tg6_xlate(struct xt_xlate * xl,const struct xt_xlate_tg_params * params)112*a71a9546SAutomerger Merge Worker static int tee_tg6_xlate(struct xt_xlate *xl,
113*a71a9546SAutomerger Merge Worker 			 const struct xt_xlate_tg_params *params)
114*a71a9546SAutomerger Merge Worker {
115*a71a9546SAutomerger Merge Worker 	const struct xt_tee_tginfo *info = (const void *)params->target->data;
116*a71a9546SAutomerger Merge Worker 
117*a71a9546SAutomerger Merge Worker 	if (params->numeric)
118*a71a9546SAutomerger Merge Worker 		xt_xlate_add(xl, "dup to %s",
119*a71a9546SAutomerger Merge Worker 			     xtables_ip6addr_to_numeric(&info->gw.in6));
120*a71a9546SAutomerger Merge Worker 	else
121*a71a9546SAutomerger Merge Worker 		xt_xlate_add(xl, "dup to %s",
122*a71a9546SAutomerger Merge Worker 			     xtables_ip6addr_to_anyname(&info->gw.in6));
123*a71a9546SAutomerger Merge Worker 	if (*info->oif != '\0')
124*a71a9546SAutomerger Merge Worker 		xt_xlate_add(xl, " device %s", info->oif);
125*a71a9546SAutomerger Merge Worker 
126*a71a9546SAutomerger Merge Worker 	return 1;
127*a71a9546SAutomerger Merge Worker }
128*a71a9546SAutomerger Merge Worker 
129*a71a9546SAutomerger Merge Worker static struct xtables_target tee_tg_reg[] = {
130*a71a9546SAutomerger Merge Worker 	{
131*a71a9546SAutomerger Merge Worker 		.name          = "TEE",
132*a71a9546SAutomerger Merge Worker 		.version       = XTABLES_VERSION,
133*a71a9546SAutomerger Merge Worker 		.revision      = 1,
134*a71a9546SAutomerger Merge Worker 		.family        = NFPROTO_IPV4,
135*a71a9546SAutomerger Merge Worker 		.size          = XT_ALIGN(sizeof(struct xt_tee_tginfo)),
136*a71a9546SAutomerger Merge Worker 		.userspacesize = offsetof(struct xt_tee_tginfo, priv),
137*a71a9546SAutomerger Merge Worker 		.help          = tee_tg_help,
138*a71a9546SAutomerger Merge Worker 		.print         = tee_tg_print,
139*a71a9546SAutomerger Merge Worker 		.save          = tee_tg_save,
140*a71a9546SAutomerger Merge Worker 		.x6_parse      = xtables_option_parse,
141*a71a9546SAutomerger Merge Worker 		.x6_options    = tee_tg_opts,
142*a71a9546SAutomerger Merge Worker 		.xlate         = tee_tg_xlate,
143*a71a9546SAutomerger Merge Worker 	},
144*a71a9546SAutomerger Merge Worker 	{
145*a71a9546SAutomerger Merge Worker 		.name          = "TEE",
146*a71a9546SAutomerger Merge Worker 		.version       = XTABLES_VERSION,
147*a71a9546SAutomerger Merge Worker 		.revision      = 1,
148*a71a9546SAutomerger Merge Worker 		.family        = NFPROTO_IPV6,
149*a71a9546SAutomerger Merge Worker 		.size          = XT_ALIGN(sizeof(struct xt_tee_tginfo)),
150*a71a9546SAutomerger Merge Worker 		.userspacesize = offsetof(struct xt_tee_tginfo, priv),
151*a71a9546SAutomerger Merge Worker 		.help          = tee_tg_help,
152*a71a9546SAutomerger Merge Worker 		.print         = tee_tg6_print,
153*a71a9546SAutomerger Merge Worker 		.save          = tee_tg6_save,
154*a71a9546SAutomerger Merge Worker 		.x6_parse      = xtables_option_parse,
155*a71a9546SAutomerger Merge Worker 		.x6_options    = tee_tg_opts,
156*a71a9546SAutomerger Merge Worker 		.xlate         = tee_tg6_xlate,
157*a71a9546SAutomerger Merge Worker 	},
158*a71a9546SAutomerger Merge Worker };
159*a71a9546SAutomerger Merge Worker 
_init(void)160*a71a9546SAutomerger Merge Worker void _init(void)
161*a71a9546SAutomerger Merge Worker {
162*a71a9546SAutomerger Merge Worker 	xtables_register_targets(tee_tg_reg, ARRAY_SIZE(tee_tg_reg));
163*a71a9546SAutomerger Merge Worker }
164