xref: /aosp_15_r20/external/iproute2/ip/link_iptnl.c (revision de1e4e894b0c224df933550f0afdecc354b238c4)
1*de1e4e89SAndroid Build Coastguard Worker /*
2*de1e4e89SAndroid Build Coastguard Worker  * link_iptnl.c	ipip and sit driver module
3*de1e4e89SAndroid Build Coastguard Worker  *
4*de1e4e89SAndroid Build Coastguard Worker  *		This program is free software; you can redistribute it and/or
5*de1e4e89SAndroid Build Coastguard Worker  *		modify it under the terms of the GNU General Public License
6*de1e4e89SAndroid Build Coastguard Worker  *		as published by the Free Software Foundation; either version
7*de1e4e89SAndroid Build Coastguard Worker  *		2 of the License, or (at your option) any later version.
8*de1e4e89SAndroid Build Coastguard Worker  *
9*de1e4e89SAndroid Build Coastguard Worker  * Authors:	Nicolas Dichtel <[email protected]>
10*de1e4e89SAndroid Build Coastguard Worker  *
11*de1e4e89SAndroid Build Coastguard Worker  */
12*de1e4e89SAndroid Build Coastguard Worker 
13*de1e4e89SAndroid Build Coastguard Worker #include <string.h>
14*de1e4e89SAndroid Build Coastguard Worker #include <net/if.h>
15*de1e4e89SAndroid Build Coastguard Worker #include <sys/types.h>
16*de1e4e89SAndroid Build Coastguard Worker #include <sys/socket.h>
17*de1e4e89SAndroid Build Coastguard Worker #include <arpa/inet.h>
18*de1e4e89SAndroid Build Coastguard Worker 
19*de1e4e89SAndroid Build Coastguard Worker #include <linux/in.h>
20*de1e4e89SAndroid Build Coastguard Worker #include <linux/ip.h>
21*de1e4e89SAndroid Build Coastguard Worker #include <linux/if_tunnel.h>
22*de1e4e89SAndroid Build Coastguard Worker #include "rt_names.h"
23*de1e4e89SAndroid Build Coastguard Worker #include "utils.h"
24*de1e4e89SAndroid Build Coastguard Worker #include "ip_common.h"
25*de1e4e89SAndroid Build Coastguard Worker #include "tunnel.h"
26*de1e4e89SAndroid Build Coastguard Worker 
print_usage(FILE * f,int sit)27*de1e4e89SAndroid Build Coastguard Worker static void print_usage(FILE *f, int sit)
28*de1e4e89SAndroid Build Coastguard Worker {
29*de1e4e89SAndroid Build Coastguard Worker 	const char *type = sit ? "sit " : "ipip";
30*de1e4e89SAndroid Build Coastguard Worker 
31*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f,
32*de1e4e89SAndroid Build Coastguard Worker 		"Usage: ... %s [ remote ADDR ]\n"
33*de1e4e89SAndroid Build Coastguard Worker 		"                [ local ADDR ]\n"
34*de1e4e89SAndroid Build Coastguard Worker 		"                [ ttl TTL ]\n"
35*de1e4e89SAndroid Build Coastguard Worker 		"                [ tos TOS ]\n"
36*de1e4e89SAndroid Build Coastguard Worker 		"                [ [no]pmtudisc ]\n"
37*de1e4e89SAndroid Build Coastguard Worker 		"                [ dev PHYS_DEV ]\n"
38*de1e4e89SAndroid Build Coastguard Worker 		"                [ 6rd-prefix ADDR ]\n"
39*de1e4e89SAndroid Build Coastguard Worker 		"                [ 6rd-relay_prefix ADDR ]\n"
40*de1e4e89SAndroid Build Coastguard Worker 		"                [ 6rd-reset ]\n"
41*de1e4e89SAndroid Build Coastguard Worker 		"                [ noencap ]\n"
42*de1e4e89SAndroid Build Coastguard Worker 		"                [ encap { fou | gue | none } ]\n"
43*de1e4e89SAndroid Build Coastguard Worker 		"                [ encap-sport PORT ]\n"
44*de1e4e89SAndroid Build Coastguard Worker 		"                [ encap-dport PORT ]\n"
45*de1e4e89SAndroid Build Coastguard Worker 		"                [ [no]encap-csum ]\n"
46*de1e4e89SAndroid Build Coastguard Worker 		"                [ [no]encap-csum6 ]\n"
47*de1e4e89SAndroid Build Coastguard Worker 		"                [ [no]encap-remcsum ]\n",
48*de1e4e89SAndroid Build Coastguard Worker 		type
49*de1e4e89SAndroid Build Coastguard Worker 	);
50*de1e4e89SAndroid Build Coastguard Worker 	if (sit) {
51*de1e4e89SAndroid Build Coastguard Worker 		fprintf(f, "          [ mode { ip6ip | ipip | mplsip | any } ]\n");
52*de1e4e89SAndroid Build Coastguard Worker 		fprintf(f, "          [ isatap ]\n");
53*de1e4e89SAndroid Build Coastguard Worker 	} else {
54*de1e4e89SAndroid Build Coastguard Worker 		fprintf(f, "          [ mode { ipip | mplsip | any } ]\n");
55*de1e4e89SAndroid Build Coastguard Worker 	}
56*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f, "                [ external ]\n");
57*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f, "                [ fwmark MARK ]\n");
58*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f, "\n");
59*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f, "Where: ADDR := { IP_ADDRESS | any }\n");
60*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f, "       TOS  := { NUMBER | inherit }\n");
61*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f, "       TTL  := { 1..255 | inherit }\n");
62*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f, "       MARK := { 0x0..0xffffffff }\n");
63*de1e4e89SAndroid Build Coastguard Worker }
64*de1e4e89SAndroid Build Coastguard Worker 
65*de1e4e89SAndroid Build Coastguard Worker static void usage(int sit) __attribute__((noreturn));
usage(int sit)66*de1e4e89SAndroid Build Coastguard Worker static void usage(int sit)
67*de1e4e89SAndroid Build Coastguard Worker {
68*de1e4e89SAndroid Build Coastguard Worker 	print_usage(stderr, sit);
69*de1e4e89SAndroid Build Coastguard Worker 	exit(-1);
70*de1e4e89SAndroid Build Coastguard Worker }
71*de1e4e89SAndroid Build Coastguard Worker 
iptunnel_parse_opt(struct link_util * lu,int argc,char ** argv,struct nlmsghdr * n)72*de1e4e89SAndroid Build Coastguard Worker static int iptunnel_parse_opt(struct link_util *lu, int argc, char **argv,
73*de1e4e89SAndroid Build Coastguard Worker 			      struct nlmsghdr *n)
74*de1e4e89SAndroid Build Coastguard Worker {
75*de1e4e89SAndroid Build Coastguard Worker 	struct ifinfomsg *ifi = (struct ifinfomsg *)(n + 1);
76*de1e4e89SAndroid Build Coastguard Worker 	struct {
77*de1e4e89SAndroid Build Coastguard Worker 		struct nlmsghdr n;
78*de1e4e89SAndroid Build Coastguard Worker 		struct ifinfomsg i;
79*de1e4e89SAndroid Build Coastguard Worker 		char buf[2048];
80*de1e4e89SAndroid Build Coastguard Worker 	} req = {
81*de1e4e89SAndroid Build Coastguard Worker 		.n.nlmsg_len = NLMSG_LENGTH(sizeof(*ifi)),
82*de1e4e89SAndroid Build Coastguard Worker 		.n.nlmsg_flags = NLM_F_REQUEST,
83*de1e4e89SAndroid Build Coastguard Worker 		.n.nlmsg_type = RTM_GETLINK,
84*de1e4e89SAndroid Build Coastguard Worker 		.i.ifi_family = preferred_family,
85*de1e4e89SAndroid Build Coastguard Worker 		.i.ifi_index = ifi->ifi_index,
86*de1e4e89SAndroid Build Coastguard Worker 	};
87*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *tb[IFLA_MAX + 1];
88*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *linkinfo[IFLA_INFO_MAX+1];
89*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *iptuninfo[IFLA_IPTUN_MAX + 1];
90*de1e4e89SAndroid Build Coastguard Worker 	int len;
91*de1e4e89SAndroid Build Coastguard Worker 	__u32 link = 0;
92*de1e4e89SAndroid Build Coastguard Worker 	__u32 laddr = 0;
93*de1e4e89SAndroid Build Coastguard Worker 	__u32 raddr = 0;
94*de1e4e89SAndroid Build Coastguard Worker 	__u8 ttl = 0;
95*de1e4e89SAndroid Build Coastguard Worker 	__u8 tos = 0;
96*de1e4e89SAndroid Build Coastguard Worker 	__u8 pmtudisc = 1;
97*de1e4e89SAndroid Build Coastguard Worker 	__u16 iflags = 0;
98*de1e4e89SAndroid Build Coastguard Worker 	__u8 proto = 0;
99*de1e4e89SAndroid Build Coastguard Worker 	struct in6_addr ip6rdprefix = {};
100*de1e4e89SAndroid Build Coastguard Worker 	__u16 ip6rdprefixlen = 0;
101*de1e4e89SAndroid Build Coastguard Worker 	__u32 ip6rdrelayprefix = 0;
102*de1e4e89SAndroid Build Coastguard Worker 	__u16 ip6rdrelayprefixlen = 0;
103*de1e4e89SAndroid Build Coastguard Worker 	__u16 encaptype = 0;
104*de1e4e89SAndroid Build Coastguard Worker 	__u16 encapflags = 0;
105*de1e4e89SAndroid Build Coastguard Worker 	__u16 encapsport = 0;
106*de1e4e89SAndroid Build Coastguard Worker 	__u16 encapdport = 0;
107*de1e4e89SAndroid Build Coastguard Worker 	__u8 metadata = 0;
108*de1e4e89SAndroid Build Coastguard Worker 	__u32 fwmark = 0;
109*de1e4e89SAndroid Build Coastguard Worker 
110*de1e4e89SAndroid Build Coastguard Worker 	if (!(n->nlmsg_flags & NLM_F_CREATE)) {
111*de1e4e89SAndroid Build Coastguard Worker 		if (rtnl_talk(&rth, &req.n, &req.n, sizeof(req)) < 0) {
112*de1e4e89SAndroid Build Coastguard Worker get_failed:
113*de1e4e89SAndroid Build Coastguard Worker 			fprintf(stderr,
114*de1e4e89SAndroid Build Coastguard Worker 				"Failed to get existing tunnel info.\n");
115*de1e4e89SAndroid Build Coastguard Worker 			return -1;
116*de1e4e89SAndroid Build Coastguard Worker 		}
117*de1e4e89SAndroid Build Coastguard Worker 
118*de1e4e89SAndroid Build Coastguard Worker 		len = req.n.nlmsg_len;
119*de1e4e89SAndroid Build Coastguard Worker 		len -= NLMSG_LENGTH(sizeof(*ifi));
120*de1e4e89SAndroid Build Coastguard Worker 		if (len < 0)
121*de1e4e89SAndroid Build Coastguard Worker 			goto get_failed;
122*de1e4e89SAndroid Build Coastguard Worker 
123*de1e4e89SAndroid Build Coastguard Worker 		parse_rtattr(tb, IFLA_MAX, IFLA_RTA(&req.i), len);
124*de1e4e89SAndroid Build Coastguard Worker 
125*de1e4e89SAndroid Build Coastguard Worker 		if (!tb[IFLA_LINKINFO])
126*de1e4e89SAndroid Build Coastguard Worker 			goto get_failed;
127*de1e4e89SAndroid Build Coastguard Worker 
128*de1e4e89SAndroid Build Coastguard Worker 		parse_rtattr_nested(linkinfo, IFLA_INFO_MAX, tb[IFLA_LINKINFO]);
129*de1e4e89SAndroid Build Coastguard Worker 
130*de1e4e89SAndroid Build Coastguard Worker 		if (!linkinfo[IFLA_INFO_DATA])
131*de1e4e89SAndroid Build Coastguard Worker 			goto get_failed;
132*de1e4e89SAndroid Build Coastguard Worker 
133*de1e4e89SAndroid Build Coastguard Worker 		parse_rtattr_nested(iptuninfo, IFLA_IPTUN_MAX,
134*de1e4e89SAndroid Build Coastguard Worker 				    linkinfo[IFLA_INFO_DATA]);
135*de1e4e89SAndroid Build Coastguard Worker 
136*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_LOCAL])
137*de1e4e89SAndroid Build Coastguard Worker 			laddr = rta_getattr_u32(iptuninfo[IFLA_IPTUN_LOCAL]);
138*de1e4e89SAndroid Build Coastguard Worker 
139*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_REMOTE])
140*de1e4e89SAndroid Build Coastguard Worker 			raddr = rta_getattr_u32(iptuninfo[IFLA_IPTUN_REMOTE]);
141*de1e4e89SAndroid Build Coastguard Worker 
142*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_TTL])
143*de1e4e89SAndroid Build Coastguard Worker 			ttl = rta_getattr_u8(iptuninfo[IFLA_IPTUN_TTL]);
144*de1e4e89SAndroid Build Coastguard Worker 
145*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_TOS])
146*de1e4e89SAndroid Build Coastguard Worker 			tos = rta_getattr_u8(iptuninfo[IFLA_IPTUN_TOS]);
147*de1e4e89SAndroid Build Coastguard Worker 
148*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_PMTUDISC])
149*de1e4e89SAndroid Build Coastguard Worker 			pmtudisc =
150*de1e4e89SAndroid Build Coastguard Worker 				rta_getattr_u8(iptuninfo[IFLA_IPTUN_PMTUDISC]);
151*de1e4e89SAndroid Build Coastguard Worker 
152*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_FLAGS])
153*de1e4e89SAndroid Build Coastguard Worker 			iflags = rta_getattr_u16(iptuninfo[IFLA_IPTUN_FLAGS]);
154*de1e4e89SAndroid Build Coastguard Worker 
155*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_LINK])
156*de1e4e89SAndroid Build Coastguard Worker 			link = rta_getattr_u32(iptuninfo[IFLA_IPTUN_LINK]);
157*de1e4e89SAndroid Build Coastguard Worker 
158*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_PROTO])
159*de1e4e89SAndroid Build Coastguard Worker 			proto = rta_getattr_u8(iptuninfo[IFLA_IPTUN_PROTO]);
160*de1e4e89SAndroid Build Coastguard Worker 
161*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_ENCAP_TYPE])
162*de1e4e89SAndroid Build Coastguard Worker 			encaptype = rta_getattr_u16(iptuninfo[IFLA_IPTUN_ENCAP_TYPE]);
163*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_ENCAP_FLAGS])
164*de1e4e89SAndroid Build Coastguard Worker 			encapflags = rta_getattr_u16(iptuninfo[IFLA_IPTUN_ENCAP_FLAGS]);
165*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_ENCAP_SPORT])
166*de1e4e89SAndroid Build Coastguard Worker 			encapsport = rta_getattr_u16(iptuninfo[IFLA_IPTUN_ENCAP_SPORT]);
167*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_ENCAP_DPORT])
168*de1e4e89SAndroid Build Coastguard Worker 			encapdport = rta_getattr_u16(iptuninfo[IFLA_IPTUN_ENCAP_DPORT]);
169*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_6RD_PREFIX])
170*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&ip6rdprefix,
171*de1e4e89SAndroid Build Coastguard Worker 			       RTA_DATA(iptuninfo[IFLA_IPTUN_6RD_PREFIX]),
172*de1e4e89SAndroid Build Coastguard Worker 			       sizeof(laddr));
173*de1e4e89SAndroid Build Coastguard Worker 
174*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_6RD_PREFIXLEN])
175*de1e4e89SAndroid Build Coastguard Worker 			ip6rdprefixlen =
176*de1e4e89SAndroid Build Coastguard Worker 				rta_getattr_u16(iptuninfo[IFLA_IPTUN_6RD_PREFIXLEN]);
177*de1e4e89SAndroid Build Coastguard Worker 
178*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_6RD_RELAY_PREFIX])
179*de1e4e89SAndroid Build Coastguard Worker 			ip6rdrelayprefix =
180*de1e4e89SAndroid Build Coastguard Worker 				rta_getattr_u32(iptuninfo[IFLA_IPTUN_6RD_RELAY_PREFIX]);
181*de1e4e89SAndroid Build Coastguard Worker 
182*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_6RD_RELAY_PREFIXLEN])
183*de1e4e89SAndroid Build Coastguard Worker 			ip6rdrelayprefixlen =
184*de1e4e89SAndroid Build Coastguard Worker 				rta_getattr_u16(iptuninfo[IFLA_IPTUN_6RD_RELAY_PREFIXLEN]);
185*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_COLLECT_METADATA])
186*de1e4e89SAndroid Build Coastguard Worker 			metadata = 1;
187*de1e4e89SAndroid Build Coastguard Worker 
188*de1e4e89SAndroid Build Coastguard Worker 		if (iptuninfo[IFLA_IPTUN_FWMARK])
189*de1e4e89SAndroid Build Coastguard Worker 			fwmark = rta_getattr_u32(iptuninfo[IFLA_IPTUN_FWMARK]);
190*de1e4e89SAndroid Build Coastguard Worker 
191*de1e4e89SAndroid Build Coastguard Worker 	}
192*de1e4e89SAndroid Build Coastguard Worker 
193*de1e4e89SAndroid Build Coastguard Worker 	while (argc > 0) {
194*de1e4e89SAndroid Build Coastguard Worker 		if (strcmp(*argv, "remote") == 0) {
195*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
196*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "any"))
197*de1e4e89SAndroid Build Coastguard Worker 				raddr = get_addr32(*argv);
198*de1e4e89SAndroid Build Coastguard Worker 			else
199*de1e4e89SAndroid Build Coastguard Worker 				raddr = 0;
200*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "local") == 0) {
201*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
202*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "any"))
203*de1e4e89SAndroid Build Coastguard Worker 				laddr = get_addr32(*argv);
204*de1e4e89SAndroid Build Coastguard Worker 			else
205*de1e4e89SAndroid Build Coastguard Worker 				laddr = 0;
206*de1e4e89SAndroid Build Coastguard Worker 		} else if (matches(*argv, "dev") == 0) {
207*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
208*de1e4e89SAndroid Build Coastguard Worker 			link = if_nametoindex(*argv);
209*de1e4e89SAndroid Build Coastguard Worker 			if (link == 0)
210*de1e4e89SAndroid Build Coastguard Worker 				invarg("\"dev\" is invalid", *argv);
211*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "ttl") == 0 ||
212*de1e4e89SAndroid Build Coastguard Worker 			   strcmp(*argv, "hoplimit") == 0) {
213*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
214*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "inherit") != 0) {
215*de1e4e89SAndroid Build Coastguard Worker 				if (get_u8(&ttl, *argv, 0))
216*de1e4e89SAndroid Build Coastguard Worker 					invarg("invalid TTL\n", *argv);
217*de1e4e89SAndroid Build Coastguard Worker 			} else
218*de1e4e89SAndroid Build Coastguard Worker 				ttl = 0;
219*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "tos") == 0 ||
220*de1e4e89SAndroid Build Coastguard Worker 			   strcmp(*argv, "tclass") == 0 ||
221*de1e4e89SAndroid Build Coastguard Worker 			   matches(*argv, "dsfield") == 0) {
222*de1e4e89SAndroid Build Coastguard Worker 			__u32 uval;
223*de1e4e89SAndroid Build Coastguard Worker 
224*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
225*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "inherit") != 0) {
226*de1e4e89SAndroid Build Coastguard Worker 				if (rtnl_dsfield_a2n(&uval, *argv))
227*de1e4e89SAndroid Build Coastguard Worker 					invarg("bad TOS value", *argv);
228*de1e4e89SAndroid Build Coastguard Worker 				tos = uval;
229*de1e4e89SAndroid Build Coastguard Worker 			} else
230*de1e4e89SAndroid Build Coastguard Worker 				tos = 1;
231*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "nopmtudisc") == 0) {
232*de1e4e89SAndroid Build Coastguard Worker 			pmtudisc = 0;
233*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "pmtudisc") == 0) {
234*de1e4e89SAndroid Build Coastguard Worker 			pmtudisc = 1;
235*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(lu->id, "sit") == 0 &&
236*de1e4e89SAndroid Build Coastguard Worker 			   strcmp(*argv, "isatap") == 0) {
237*de1e4e89SAndroid Build Coastguard Worker 			iflags |= SIT_ISATAP;
238*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(lu->id, "sit") == 0 &&
239*de1e4e89SAndroid Build Coastguard Worker 			   strcmp(*argv, "mode") == 0) {
240*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
241*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "ipv6/ipv4") == 0 ||
242*de1e4e89SAndroid Build Coastguard Worker 			    strcmp(*argv, "ip6ip") == 0)
243*de1e4e89SAndroid Build Coastguard Worker 				proto = IPPROTO_IPV6;
244*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "ipv4/ipv4") == 0 ||
245*de1e4e89SAndroid Build Coastguard Worker 				 strcmp(*argv, "ipip") == 0 ||
246*de1e4e89SAndroid Build Coastguard Worker 				 strcmp(*argv, "ip4ip4") == 0)
247*de1e4e89SAndroid Build Coastguard Worker 				proto = IPPROTO_IPIP;
248*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "mpls/ipv4") == 0 ||
249*de1e4e89SAndroid Build Coastguard Worker 				   strcmp(*argv, "mplsip") == 0)
250*de1e4e89SAndroid Build Coastguard Worker 				proto = IPPROTO_MPLS;
251*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "any/ipv4") == 0 ||
252*de1e4e89SAndroid Build Coastguard Worker 				 strcmp(*argv, "any") == 0)
253*de1e4e89SAndroid Build Coastguard Worker 				proto = 0;
254*de1e4e89SAndroid Build Coastguard Worker 			else
255*de1e4e89SAndroid Build Coastguard Worker 				invarg("Cannot guess tunnel mode.", *argv);
256*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(lu->id, "ipip") == 0 &&
257*de1e4e89SAndroid Build Coastguard Worker 			   strcmp(*argv, "mode") == 0) {
258*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
259*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "ipv4/ipv4") == 0 ||
260*de1e4e89SAndroid Build Coastguard Worker 				 strcmp(*argv, "ipip") == 0 ||
261*de1e4e89SAndroid Build Coastguard Worker 				 strcmp(*argv, "ip4ip4") == 0)
262*de1e4e89SAndroid Build Coastguard Worker 				proto = IPPROTO_IPIP;
263*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "mpls/ipv4") == 0 ||
264*de1e4e89SAndroid Build Coastguard Worker 				   strcmp(*argv, "mplsip") == 0)
265*de1e4e89SAndroid Build Coastguard Worker 				proto = IPPROTO_MPLS;
266*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "any/ipv4") == 0 ||
267*de1e4e89SAndroid Build Coastguard Worker 				 strcmp(*argv, "any") == 0)
268*de1e4e89SAndroid Build Coastguard Worker 				proto = 0;
269*de1e4e89SAndroid Build Coastguard Worker 			else
270*de1e4e89SAndroid Build Coastguard Worker 				invarg("Cannot guess tunnel mode.", *argv);
271*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap") == 0) {
272*de1e4e89SAndroid Build Coastguard Worker 			encaptype = TUNNEL_ENCAP_NONE;
273*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap") == 0) {
274*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
275*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "fou") == 0)
276*de1e4e89SAndroid Build Coastguard Worker 				encaptype = TUNNEL_ENCAP_FOU;
277*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "gue") == 0)
278*de1e4e89SAndroid Build Coastguard Worker 				encaptype = TUNNEL_ENCAP_GUE;
279*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "none") == 0)
280*de1e4e89SAndroid Build Coastguard Worker 				encaptype = TUNNEL_ENCAP_NONE;
281*de1e4e89SAndroid Build Coastguard Worker 			else
282*de1e4e89SAndroid Build Coastguard Worker 				invarg("Invalid encap type.", *argv);
283*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-sport") == 0) {
284*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
285*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "auto") == 0)
286*de1e4e89SAndroid Build Coastguard Worker 				encapsport = 0;
287*de1e4e89SAndroid Build Coastguard Worker 			else if (get_u16(&encapsport, *argv, 0))
288*de1e4e89SAndroid Build Coastguard Worker 				invarg("Invalid source port.", *argv);
289*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-dport") == 0) {
290*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
291*de1e4e89SAndroid Build Coastguard Worker 			if (get_u16(&encapdport, *argv, 0))
292*de1e4e89SAndroid Build Coastguard Worker 				invarg("Invalid destination port.", *argv);
293*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-csum") == 0) {
294*de1e4e89SAndroid Build Coastguard Worker 			encapflags |= TUNNEL_ENCAP_FLAG_CSUM;
295*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap-csum") == 0) {
296*de1e4e89SAndroid Build Coastguard Worker 			encapflags &= ~TUNNEL_ENCAP_FLAG_CSUM;
297*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-udp6-csum") == 0) {
298*de1e4e89SAndroid Build Coastguard Worker 			encapflags |= TUNNEL_ENCAP_FLAG_CSUM6;
299*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap-udp6-csum") == 0) {
300*de1e4e89SAndroid Build Coastguard Worker 			encapflags &= ~TUNNEL_ENCAP_FLAG_CSUM6;
301*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-remcsum") == 0) {
302*de1e4e89SAndroid Build Coastguard Worker 			encapflags |= TUNNEL_ENCAP_FLAG_REMCSUM;
303*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap-remcsum") == 0) {
304*de1e4e89SAndroid Build Coastguard Worker 			encapflags &= ~TUNNEL_ENCAP_FLAG_REMCSUM;
305*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "external") == 0) {
306*de1e4e89SAndroid Build Coastguard Worker 			metadata = 1;
307*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "6rd-prefix") == 0) {
308*de1e4e89SAndroid Build Coastguard Worker 			inet_prefix prefix;
309*de1e4e89SAndroid Build Coastguard Worker 
310*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
311*de1e4e89SAndroid Build Coastguard Worker 			if (get_prefix(&prefix, *argv, AF_INET6))
312*de1e4e89SAndroid Build Coastguard Worker 				invarg("invalid 6rd_prefix\n", *argv);
313*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&ip6rdprefix, prefix.data, 16);
314*de1e4e89SAndroid Build Coastguard Worker 			ip6rdprefixlen = prefix.bitlen;
315*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "6rd-relay_prefix") == 0) {
316*de1e4e89SAndroid Build Coastguard Worker 			inet_prefix prefix;
317*de1e4e89SAndroid Build Coastguard Worker 
318*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
319*de1e4e89SAndroid Build Coastguard Worker 			if (get_prefix(&prefix, *argv, AF_INET))
320*de1e4e89SAndroid Build Coastguard Worker 				invarg("invalid 6rd-relay_prefix\n", *argv);
321*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&ip6rdrelayprefix, prefix.data, 4);
322*de1e4e89SAndroid Build Coastguard Worker 			ip6rdrelayprefixlen = prefix.bitlen;
323*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "6rd-reset") == 0) {
324*de1e4e89SAndroid Build Coastguard Worker 			inet_prefix prefix;
325*de1e4e89SAndroid Build Coastguard Worker 
326*de1e4e89SAndroid Build Coastguard Worker 			get_prefix(&prefix, "2002::", AF_INET6);
327*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&ip6rdprefix, prefix.data, 16);
328*de1e4e89SAndroid Build Coastguard Worker 			ip6rdprefixlen = 16;
329*de1e4e89SAndroid Build Coastguard Worker 			ip6rdrelayprefix = 0;
330*de1e4e89SAndroid Build Coastguard Worker 			ip6rdrelayprefixlen = 0;
331*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "fwmark") == 0) {
332*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
333*de1e4e89SAndroid Build Coastguard Worker 			if (get_u32(&fwmark, *argv, 0))
334*de1e4e89SAndroid Build Coastguard Worker 				invarg("invalid fwmark\n", *argv);
335*de1e4e89SAndroid Build Coastguard Worker 		} else
336*de1e4e89SAndroid Build Coastguard Worker 			usage(strcmp(lu->id, "sit") == 0);
337*de1e4e89SAndroid Build Coastguard Worker 		argc--, argv++;
338*de1e4e89SAndroid Build Coastguard Worker 	}
339*de1e4e89SAndroid Build Coastguard Worker 
340*de1e4e89SAndroid Build Coastguard Worker 	if (ttl && pmtudisc == 0) {
341*de1e4e89SAndroid Build Coastguard Worker 		fprintf(stderr, "ttl != 0 and nopmtudisc are incompatible\n");
342*de1e4e89SAndroid Build Coastguard Worker 		exit(-1);
343*de1e4e89SAndroid Build Coastguard Worker 	}
344*de1e4e89SAndroid Build Coastguard Worker 
345*de1e4e89SAndroid Build Coastguard Worker 	if (metadata) {
346*de1e4e89SAndroid Build Coastguard Worker 		addattr_l(n, 1024, IFLA_IPTUN_COLLECT_METADATA, NULL, 0);
347*de1e4e89SAndroid Build Coastguard Worker 		return 0;
348*de1e4e89SAndroid Build Coastguard Worker 	}
349*de1e4e89SAndroid Build Coastguard Worker 
350*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_IPTUN_LINK, link);
351*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_IPTUN_LOCAL, laddr);
352*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_IPTUN_REMOTE, raddr);
353*de1e4e89SAndroid Build Coastguard Worker 	addattr8(n, 1024, IFLA_IPTUN_TTL, ttl);
354*de1e4e89SAndroid Build Coastguard Worker 	addattr8(n, 1024, IFLA_IPTUN_TOS, tos);
355*de1e4e89SAndroid Build Coastguard Worker 	addattr8(n, 1024, IFLA_IPTUN_PMTUDISC, pmtudisc);
356*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_IPTUN_FWMARK, fwmark);
357*de1e4e89SAndroid Build Coastguard Worker 
358*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_IPTUN_ENCAP_TYPE, encaptype);
359*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_IPTUN_ENCAP_FLAGS, encapflags);
360*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_IPTUN_ENCAP_SPORT, htons(encapsport));
361*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_IPTUN_ENCAP_DPORT, htons(encapdport));
362*de1e4e89SAndroid Build Coastguard Worker 
363*de1e4e89SAndroid Build Coastguard Worker 	if (strcmp(lu->id, "ipip") == 0 || strcmp(lu->id, "sit") == 0)
364*de1e4e89SAndroid Build Coastguard Worker 		addattr8(n, 1024, IFLA_IPTUN_PROTO, proto);
365*de1e4e89SAndroid Build Coastguard Worker 
366*de1e4e89SAndroid Build Coastguard Worker 	if (strcmp(lu->id, "sit") == 0) {
367*de1e4e89SAndroid Build Coastguard Worker 		addattr16(n, 1024, IFLA_IPTUN_FLAGS, iflags);
368*de1e4e89SAndroid Build Coastguard Worker 		if (ip6rdprefixlen) {
369*de1e4e89SAndroid Build Coastguard Worker 			addattr_l(n, 1024, IFLA_IPTUN_6RD_PREFIX,
370*de1e4e89SAndroid Build Coastguard Worker 				  &ip6rdprefix, sizeof(ip6rdprefix));
371*de1e4e89SAndroid Build Coastguard Worker 			addattr16(n, 1024, IFLA_IPTUN_6RD_PREFIXLEN,
372*de1e4e89SAndroid Build Coastguard Worker 				  ip6rdprefixlen);
373*de1e4e89SAndroid Build Coastguard Worker 			addattr32(n, 1024, IFLA_IPTUN_6RD_RELAY_PREFIX,
374*de1e4e89SAndroid Build Coastguard Worker 				  ip6rdrelayprefix);
375*de1e4e89SAndroid Build Coastguard Worker 			addattr16(n, 1024, IFLA_IPTUN_6RD_RELAY_PREFIXLEN,
376*de1e4e89SAndroid Build Coastguard Worker 				  ip6rdrelayprefixlen);
377*de1e4e89SAndroid Build Coastguard Worker 		}
378*de1e4e89SAndroid Build Coastguard Worker 	}
379*de1e4e89SAndroid Build Coastguard Worker 
380*de1e4e89SAndroid Build Coastguard Worker 	return 0;
381*de1e4e89SAndroid Build Coastguard Worker }
382*de1e4e89SAndroid Build Coastguard Worker 
iptunnel_print_opt(struct link_util * lu,FILE * f,struct rtattr * tb[])383*de1e4e89SAndroid Build Coastguard Worker static void iptunnel_print_opt(struct link_util *lu, FILE *f, struct rtattr *tb[])
384*de1e4e89SAndroid Build Coastguard Worker {
385*de1e4e89SAndroid Build Coastguard Worker 	char s1[1024];
386*de1e4e89SAndroid Build Coastguard Worker 	char s2[64];
387*de1e4e89SAndroid Build Coastguard Worker 	const char *local = "any";
388*de1e4e89SAndroid Build Coastguard Worker 	const char *remote = "any";
389*de1e4e89SAndroid Build Coastguard Worker 	__u16 prefixlen, type;
390*de1e4e89SAndroid Build Coastguard Worker 
391*de1e4e89SAndroid Build Coastguard Worker 	if (!tb)
392*de1e4e89SAndroid Build Coastguard Worker 		return;
393*de1e4e89SAndroid Build Coastguard Worker 
394*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_REMOTE]) {
395*de1e4e89SAndroid Build Coastguard Worker 		unsigned int addr = rta_getattr_u32(tb[IFLA_IPTUN_REMOTE]);
396*de1e4e89SAndroid Build Coastguard Worker 
397*de1e4e89SAndroid Build Coastguard Worker 		if (addr)
398*de1e4e89SAndroid Build Coastguard Worker 			remote = format_host(AF_INET, 4, &addr);
399*de1e4e89SAndroid Build Coastguard Worker 	}
400*de1e4e89SAndroid Build Coastguard Worker 
401*de1e4e89SAndroid Build Coastguard Worker 	print_string(PRINT_ANY, "remote", "remote %s ", remote);
402*de1e4e89SAndroid Build Coastguard Worker 
403*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_LOCAL]) {
404*de1e4e89SAndroid Build Coastguard Worker 		unsigned int addr = rta_getattr_u32(tb[IFLA_IPTUN_LOCAL]);
405*de1e4e89SAndroid Build Coastguard Worker 
406*de1e4e89SAndroid Build Coastguard Worker 		if (addr)
407*de1e4e89SAndroid Build Coastguard Worker 			local = format_host(AF_INET, 4, &addr);
408*de1e4e89SAndroid Build Coastguard Worker 	}
409*de1e4e89SAndroid Build Coastguard Worker 
410*de1e4e89SAndroid Build Coastguard Worker 	print_string(PRINT_ANY, "local", "local %s ", local);
411*de1e4e89SAndroid Build Coastguard Worker 
412*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_LINK] && rta_getattr_u32(tb[IFLA_IPTUN_LINK])) {
413*de1e4e89SAndroid Build Coastguard Worker 		unsigned int link = rta_getattr_u32(tb[IFLA_IPTUN_LINK]);
414*de1e4e89SAndroid Build Coastguard Worker 		const char *n = if_indextoname(link, s2);
415*de1e4e89SAndroid Build Coastguard Worker 
416*de1e4e89SAndroid Build Coastguard Worker 		if (n)
417*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "link", "dev %s ", n);
418*de1e4e89SAndroid Build Coastguard Worker 		else
419*de1e4e89SAndroid Build Coastguard Worker 			print_int(PRINT_ANY, "link_index", "dev %u ", link);
420*de1e4e89SAndroid Build Coastguard Worker 	}
421*de1e4e89SAndroid Build Coastguard Worker 
422*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_TTL]) {
423*de1e4e89SAndroid Build Coastguard Worker 		__u8 ttl = rta_getattr_u8(tb[IFLA_IPTUN_TTL]);
424*de1e4e89SAndroid Build Coastguard Worker 
425*de1e4e89SAndroid Build Coastguard Worker 		if (ttl)
426*de1e4e89SAndroid Build Coastguard Worker 			print_int(PRINT_ANY, "ttl", "ttl %d ", ttl);
427*de1e4e89SAndroid Build Coastguard Worker 		else
428*de1e4e89SAndroid Build Coastguard Worker 			print_int(PRINT_JSON, "ttl", NULL, ttl);
429*de1e4e89SAndroid Build Coastguard Worker 	} else {
430*de1e4e89SAndroid Build Coastguard Worker 		print_string(PRINT_FP, NULL, "ttl %s ", "inherit");
431*de1e4e89SAndroid Build Coastguard Worker 	}
432*de1e4e89SAndroid Build Coastguard Worker 
433*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_TOS]) {
434*de1e4e89SAndroid Build Coastguard Worker 		int tos = rta_getattr_u8(tb[IFLA_IPTUN_TOS]);
435*de1e4e89SAndroid Build Coastguard Worker 
436*de1e4e89SAndroid Build Coastguard Worker 		if (tos) {
437*de1e4e89SAndroid Build Coastguard Worker 			if (is_json_context()) {
438*de1e4e89SAndroid Build Coastguard Worker 				print_0xhex(PRINT_JSON, "tos", "%#x", tos);
439*de1e4e89SAndroid Build Coastguard Worker 			} else {
440*de1e4e89SAndroid Build Coastguard Worker 				fputs("tos ", f);
441*de1e4e89SAndroid Build Coastguard Worker 				if (tos == 1)
442*de1e4e89SAndroid Build Coastguard Worker 					fputs("inherit ", f);
443*de1e4e89SAndroid Build Coastguard Worker 				else
444*de1e4e89SAndroid Build Coastguard Worker 					fprintf(f, "0x%x ", tos);
445*de1e4e89SAndroid Build Coastguard Worker 			}
446*de1e4e89SAndroid Build Coastguard Worker 		}
447*de1e4e89SAndroid Build Coastguard Worker 	}
448*de1e4e89SAndroid Build Coastguard Worker 
449*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_PMTUDISC] && rta_getattr_u8(tb[IFLA_IPTUN_PMTUDISC]))
450*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY, "pmtudisc", "pmtudisc ", true);
451*de1e4e89SAndroid Build Coastguard Worker 	else
452*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY, "pmtudisc", "nopmtudisc ", false);
453*de1e4e89SAndroid Build Coastguard Worker 
454*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_FLAGS]) {
455*de1e4e89SAndroid Build Coastguard Worker 		__u16 iflags = rta_getattr_u16(tb[IFLA_IPTUN_FLAGS]);
456*de1e4e89SAndroid Build Coastguard Worker 
457*de1e4e89SAndroid Build Coastguard Worker 		if (iflags & SIT_ISATAP)
458*de1e4e89SAndroid Build Coastguard Worker 			print_bool(PRINT_ANY, "isatap", "isatap ", true);
459*de1e4e89SAndroid Build Coastguard Worker 	}
460*de1e4e89SAndroid Build Coastguard Worker 
461*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_6RD_PREFIXLEN] &&
462*de1e4e89SAndroid Build Coastguard Worker 	    (prefixlen = rta_getattr_u16(tb[IFLA_IPTUN_6RD_PREFIXLEN]))) {
463*de1e4e89SAndroid Build Coastguard Worker 		__u16 relayprefixlen =
464*de1e4e89SAndroid Build Coastguard Worker 			rta_getattr_u16(tb[IFLA_IPTUN_6RD_RELAY_PREFIXLEN]);
465*de1e4e89SAndroid Build Coastguard Worker 		__u32 relayprefix =
466*de1e4e89SAndroid Build Coastguard Worker 			rta_getattr_u32(tb[IFLA_IPTUN_6RD_RELAY_PREFIX]);
467*de1e4e89SAndroid Build Coastguard Worker 
468*de1e4e89SAndroid Build Coastguard Worker 		const char *prefix = inet_ntop(AF_INET6,
469*de1e4e89SAndroid Build Coastguard Worker 					       RTA_DATA(tb[IFLA_IPTUN_6RD_PREFIX]),
470*de1e4e89SAndroid Build Coastguard Worker 					       s1, sizeof(s1));
471*de1e4e89SAndroid Build Coastguard Worker 
472*de1e4e89SAndroid Build Coastguard Worker 		if (is_json_context()) {
473*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_JSON, "prefix", NULL, prefix);
474*de1e4e89SAndroid Build Coastguard Worker 			print_int(PRINT_JSON, "prefixlen", NULL, prefixlen);
475*de1e4e89SAndroid Build Coastguard Worker 			if (relayprefix) {
476*de1e4e89SAndroid Build Coastguard Worker 				print_string(PRINT_JSON,
477*de1e4e89SAndroid Build Coastguard Worker 					     "relay_prefix",
478*de1e4e89SAndroid Build Coastguard Worker 					     NULL,
479*de1e4e89SAndroid Build Coastguard Worker 					     format_host(AF_INET,
480*de1e4e89SAndroid Build Coastguard Worker 							 4,
481*de1e4e89SAndroid Build Coastguard Worker 							 &relayprefix));
482*de1e4e89SAndroid Build Coastguard Worker 				print_int(PRINT_JSON,
483*de1e4e89SAndroid Build Coastguard Worker 					  "relay_prefixlen",
484*de1e4e89SAndroid Build Coastguard Worker 					  NULL,
485*de1e4e89SAndroid Build Coastguard Worker 					  relayprefixlen);
486*de1e4e89SAndroid Build Coastguard Worker 			}
487*de1e4e89SAndroid Build Coastguard Worker 		} else {
488*de1e4e89SAndroid Build Coastguard Worker 			printf("6rd-prefix %s/%u ", prefix, prefixlen);
489*de1e4e89SAndroid Build Coastguard Worker 			if (relayprefix) {
490*de1e4e89SAndroid Build Coastguard Worker 				printf("6rd-relay_prefix %s/%u ",
491*de1e4e89SAndroid Build Coastguard Worker 				       format_host(AF_INET, 4, &relayprefix),
492*de1e4e89SAndroid Build Coastguard Worker 				       relayprefixlen);
493*de1e4e89SAndroid Build Coastguard Worker 			}
494*de1e4e89SAndroid Build Coastguard Worker 		}
495*de1e4e89SAndroid Build Coastguard Worker 	}
496*de1e4e89SAndroid Build Coastguard Worker 
497*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_ENCAP_TYPE] &&
498*de1e4e89SAndroid Build Coastguard Worker 	    (type = rta_getattr_u16(tb[IFLA_IPTUN_ENCAP_TYPE])) != TUNNEL_ENCAP_NONE) {
499*de1e4e89SAndroid Build Coastguard Worker 		__u16 flags = rta_getattr_u16(tb[IFLA_IPTUN_ENCAP_FLAGS]);
500*de1e4e89SAndroid Build Coastguard Worker 		__u16 sport = rta_getattr_u16(tb[IFLA_IPTUN_ENCAP_SPORT]);
501*de1e4e89SAndroid Build Coastguard Worker 		__u16 dport = rta_getattr_u16(tb[IFLA_IPTUN_ENCAP_DPORT]);
502*de1e4e89SAndroid Build Coastguard Worker 
503*de1e4e89SAndroid Build Coastguard Worker 		print_string(PRINT_FP, NULL, "encap ", NULL);
504*de1e4e89SAndroid Build Coastguard Worker 		switch (type) {
505*de1e4e89SAndroid Build Coastguard Worker 		case TUNNEL_ENCAP_FOU:
506*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "type", "%s ", "fou");
507*de1e4e89SAndroid Build Coastguard Worker 			break;
508*de1e4e89SAndroid Build Coastguard Worker 		case TUNNEL_ENCAP_GUE:
509*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "type", "%s ", "gue");
510*de1e4e89SAndroid Build Coastguard Worker 			break;
511*de1e4e89SAndroid Build Coastguard Worker 		default:
512*de1e4e89SAndroid Build Coastguard Worker 			print_null(PRINT_ANY, "type", "unknown ", NULL);
513*de1e4e89SAndroid Build Coastguard Worker 			break;
514*de1e4e89SAndroid Build Coastguard Worker 		}
515*de1e4e89SAndroid Build Coastguard Worker 
516*de1e4e89SAndroid Build Coastguard Worker 		if (is_json_context()) {
517*de1e4e89SAndroid Build Coastguard Worker 			print_uint(PRINT_JSON,
518*de1e4e89SAndroid Build Coastguard Worker 				   "sport",
519*de1e4e89SAndroid Build Coastguard Worker 				   NULL,
520*de1e4e89SAndroid Build Coastguard Worker 				   sport ? ntohs(sport) : 0);
521*de1e4e89SAndroid Build Coastguard Worker 			print_uint(PRINT_JSON, "dport", NULL, ntohs(dport));
522*de1e4e89SAndroid Build Coastguard Worker 			print_bool(PRINT_JSON,
523*de1e4e89SAndroid Build Coastguard Worker 				   "csum",
524*de1e4e89SAndroid Build Coastguard Worker 				   NULL,
525*de1e4e89SAndroid Build Coastguard Worker 				   flags & TUNNEL_ENCAP_FLAG_CSUM);
526*de1e4e89SAndroid Build Coastguard Worker 			print_bool(PRINT_JSON,
527*de1e4e89SAndroid Build Coastguard Worker 				   "csum6",
528*de1e4e89SAndroid Build Coastguard Worker 				   NULL,
529*de1e4e89SAndroid Build Coastguard Worker 				   flags & TUNNEL_ENCAP_FLAG_CSUM6);
530*de1e4e89SAndroid Build Coastguard Worker 			print_bool(PRINT_JSON,
531*de1e4e89SAndroid Build Coastguard Worker 				   "remcsum",
532*de1e4e89SAndroid Build Coastguard Worker 				   NULL,
533*de1e4e89SAndroid Build Coastguard Worker 				   flags & TUNNEL_ENCAP_FLAG_REMCSUM);
534*de1e4e89SAndroid Build Coastguard Worker 			close_json_object();
535*de1e4e89SAndroid Build Coastguard Worker 		} else {
536*de1e4e89SAndroid Build Coastguard Worker 			if (sport == 0)
537*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-sport auto ", f);
538*de1e4e89SAndroid Build Coastguard Worker 			else
539*de1e4e89SAndroid Build Coastguard Worker 				fprintf(f, "encap-sport %u", ntohs(sport));
540*de1e4e89SAndroid Build Coastguard Worker 
541*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "encap-dport %u ", ntohs(dport));
542*de1e4e89SAndroid Build Coastguard Worker 
543*de1e4e89SAndroid Build Coastguard Worker 			if (flags & TUNNEL_ENCAP_FLAG_CSUM)
544*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-csum ", f);
545*de1e4e89SAndroid Build Coastguard Worker 			else
546*de1e4e89SAndroid Build Coastguard Worker 				fputs("noencap-csum ", f);
547*de1e4e89SAndroid Build Coastguard Worker 
548*de1e4e89SAndroid Build Coastguard Worker 			if (flags & TUNNEL_ENCAP_FLAG_CSUM6)
549*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-csum6 ", f);
550*de1e4e89SAndroid Build Coastguard Worker 			else
551*de1e4e89SAndroid Build Coastguard Worker 				fputs("noencap-csum6 ", f);
552*de1e4e89SAndroid Build Coastguard Worker 
553*de1e4e89SAndroid Build Coastguard Worker 			if (flags & TUNNEL_ENCAP_FLAG_REMCSUM)
554*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-remcsum ", f);
555*de1e4e89SAndroid Build Coastguard Worker 			else
556*de1e4e89SAndroid Build Coastguard Worker 				fputs("noencap-remcsum ", f);
557*de1e4e89SAndroid Build Coastguard Worker 		}
558*de1e4e89SAndroid Build Coastguard Worker 	}
559*de1e4e89SAndroid Build Coastguard Worker 
560*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_IPTUN_FWMARK]) {
561*de1e4e89SAndroid Build Coastguard Worker 		__u32 fwmark = rta_getattr_u32(tb[IFLA_IPTUN_FWMARK]);
562*de1e4e89SAndroid Build Coastguard Worker 
563*de1e4e89SAndroid Build Coastguard Worker 		if (fwmark) {
564*de1e4e89SAndroid Build Coastguard Worker 			snprintf(s2, sizeof(s2), "0x%x", fwmark);
565*de1e4e89SAndroid Build Coastguard Worker 
566*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "fwmark", "fwmark %s ", s2);
567*de1e4e89SAndroid Build Coastguard Worker 		}
568*de1e4e89SAndroid Build Coastguard Worker 	}
569*de1e4e89SAndroid Build Coastguard Worker }
570*de1e4e89SAndroid Build Coastguard Worker 
iptunnel_print_help(struct link_util * lu,int argc,char ** argv,FILE * f)571*de1e4e89SAndroid Build Coastguard Worker static void iptunnel_print_help(struct link_util *lu, int argc, char **argv,
572*de1e4e89SAndroid Build Coastguard Worker 	FILE *f)
573*de1e4e89SAndroid Build Coastguard Worker {
574*de1e4e89SAndroid Build Coastguard Worker 	print_usage(f, strcmp(lu->id, "sit") == 0);
575*de1e4e89SAndroid Build Coastguard Worker }
576*de1e4e89SAndroid Build Coastguard Worker 
577*de1e4e89SAndroid Build Coastguard Worker struct link_util ipip_link_util = {
578*de1e4e89SAndroid Build Coastguard Worker 	.id = "ipip",
579*de1e4e89SAndroid Build Coastguard Worker 	.maxattr = IFLA_IPTUN_MAX,
580*de1e4e89SAndroid Build Coastguard Worker 	.parse_opt = iptunnel_parse_opt,
581*de1e4e89SAndroid Build Coastguard Worker 	.print_opt = iptunnel_print_opt,
582*de1e4e89SAndroid Build Coastguard Worker 	.print_help = iptunnel_print_help,
583*de1e4e89SAndroid Build Coastguard Worker };
584*de1e4e89SAndroid Build Coastguard Worker 
585*de1e4e89SAndroid Build Coastguard Worker struct link_util sit_link_util = {
586*de1e4e89SAndroid Build Coastguard Worker 	.id = "sit",
587*de1e4e89SAndroid Build Coastguard Worker 	.maxattr = IFLA_IPTUN_MAX,
588*de1e4e89SAndroid Build Coastguard Worker 	.parse_opt = iptunnel_parse_opt,
589*de1e4e89SAndroid Build Coastguard Worker 	.print_opt = iptunnel_print_opt,
590*de1e4e89SAndroid Build Coastguard Worker 	.print_help = iptunnel_print_help,
591*de1e4e89SAndroid Build Coastguard Worker };
592