xref: /aosp_15_r20/external/iproute2/ip/link_gre6.c (revision de1e4e894b0c224df933550f0afdecc354b238c4)
1*de1e4e89SAndroid Build Coastguard Worker /*
2*de1e4e89SAndroid Build Coastguard Worker  * link_gre6.c	gre 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:	Dmitry Kozlov <[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/ip.h>
20*de1e4e89SAndroid Build Coastguard Worker #include <linux/if_tunnel.h>
21*de1e4e89SAndroid Build Coastguard Worker #include <linux/ip6_tunnel.h>
22*de1e4e89SAndroid Build Coastguard Worker 
23*de1e4e89SAndroid Build Coastguard Worker #include "rt_names.h"
24*de1e4e89SAndroid Build Coastguard Worker #include "utils.h"
25*de1e4e89SAndroid Build Coastguard Worker #include "ip_common.h"
26*de1e4e89SAndroid Build Coastguard Worker #include "tunnel.h"
27*de1e4e89SAndroid Build Coastguard Worker 
28*de1e4e89SAndroid Build Coastguard Worker #define IP6_FLOWINFO_TCLASS	htonl(0x0FF00000)
29*de1e4e89SAndroid Build Coastguard Worker #define IP6_FLOWINFO_FLOWLABEL	htonl(0x000FFFFF)
30*de1e4e89SAndroid Build Coastguard Worker 
31*de1e4e89SAndroid Build Coastguard Worker #define DEFAULT_TNL_HOP_LIMIT	(64)
32*de1e4e89SAndroid Build Coastguard Worker 
print_usage(FILE * f)33*de1e4e89SAndroid Build Coastguard Worker static void print_usage(FILE *f)
34*de1e4e89SAndroid Build Coastguard Worker {
35*de1e4e89SAndroid Build Coastguard Worker 	fprintf(f,
36*de1e4e89SAndroid Build Coastguard Worker 		"Usage: ... { ip6gre | ip6gretap } [ remote ADDR ]\n"
37*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ local ADDR ]\n"
38*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ [i|o]seq ]\n"
39*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ [i|o]key KEY ]\n"
40*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ [i|o]csum ]\n"
41*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ hoplimit TTL ]\n"
42*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ encaplimit ELIM ]\n"
43*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ tclass TCLASS ]\n"
44*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ flowlabel FLOWLABEL ]\n"
45*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ dscp inherit ]\n"
46*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ fwmark MARK ]\n"
47*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ dev PHYS_DEV ]\n"
48*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ noencap ]\n"
49*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ encap { fou | gue | none } ]\n"
50*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ encap-sport PORT ]\n"
51*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ encap-dport PORT ]\n"
52*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ [no]encap-csum ]\n"
53*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ [no]encap-csum6 ]\n"
54*de1e4e89SAndroid Build Coastguard Worker 		"                                  [ [no]encap-remcsum ]\n"
55*de1e4e89SAndroid Build Coastguard Worker 		"\n"
56*de1e4e89SAndroid Build Coastguard Worker 		"Where: ADDR      := IPV6_ADDRESS\n"
57*de1e4e89SAndroid Build Coastguard Worker 		"       TTL       := { 0..255 } (default=%d)\n"
58*de1e4e89SAndroid Build Coastguard Worker 		"       KEY       := { DOTTED_QUAD | NUMBER }\n"
59*de1e4e89SAndroid Build Coastguard Worker 		"       ELIM      := { none | 0..255 }(default=%d)\n"
60*de1e4e89SAndroid Build Coastguard Worker 		"       TCLASS    := { 0x0..0xff | inherit }\n"
61*de1e4e89SAndroid Build Coastguard Worker 		"       FLOWLABEL := { 0x0..0xfffff | inherit }\n"
62*de1e4e89SAndroid Build Coastguard Worker 		"       MARK      := { 0x0..0xffffffff | inherit }\n",
63*de1e4e89SAndroid Build Coastguard Worker 		DEFAULT_TNL_HOP_LIMIT, IPV6_DEFAULT_TNL_ENCAP_LIMIT
64*de1e4e89SAndroid Build Coastguard Worker 	);
65*de1e4e89SAndroid Build Coastguard Worker }
66*de1e4e89SAndroid Build Coastguard Worker 
67*de1e4e89SAndroid Build Coastguard Worker static void usage(void) __attribute__((noreturn));
usage(void)68*de1e4e89SAndroid Build Coastguard Worker static void usage(void)
69*de1e4e89SAndroid Build Coastguard Worker {
70*de1e4e89SAndroid Build Coastguard Worker 	print_usage(stderr);
71*de1e4e89SAndroid Build Coastguard Worker 	exit(-1);
72*de1e4e89SAndroid Build Coastguard Worker }
73*de1e4e89SAndroid Build Coastguard Worker 
gre_parse_opt(struct link_util * lu,int argc,char ** argv,struct nlmsghdr * n)74*de1e4e89SAndroid Build Coastguard Worker static int gre_parse_opt(struct link_util *lu, int argc, char **argv,
75*de1e4e89SAndroid Build Coastguard Worker 			 struct nlmsghdr *n)
76*de1e4e89SAndroid Build Coastguard Worker {
77*de1e4e89SAndroid Build Coastguard Worker 	struct ifinfomsg *ifi = (struct ifinfomsg *)(n + 1);
78*de1e4e89SAndroid Build Coastguard Worker 	struct {
79*de1e4e89SAndroid Build Coastguard Worker 		struct nlmsghdr n;
80*de1e4e89SAndroid Build Coastguard Worker 		struct ifinfomsg i;
81*de1e4e89SAndroid Build Coastguard Worker 		char buf[1024];
82*de1e4e89SAndroid Build Coastguard Worker 	} req = {
83*de1e4e89SAndroid Build Coastguard Worker 		.n.nlmsg_len = NLMSG_LENGTH(sizeof(*ifi)),
84*de1e4e89SAndroid Build Coastguard Worker 		.n.nlmsg_flags = NLM_F_REQUEST,
85*de1e4e89SAndroid Build Coastguard Worker 		.n.nlmsg_type = RTM_GETLINK,
86*de1e4e89SAndroid Build Coastguard Worker 		.i.ifi_family = preferred_family,
87*de1e4e89SAndroid Build Coastguard Worker 		.i.ifi_index = ifi->ifi_index,
88*de1e4e89SAndroid Build Coastguard Worker 	};
89*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *tb[IFLA_MAX + 1];
90*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *linkinfo[IFLA_INFO_MAX+1];
91*de1e4e89SAndroid Build Coastguard Worker 	struct rtattr *greinfo[IFLA_GRE_MAX + 1];
92*de1e4e89SAndroid Build Coastguard Worker 	__u16 iflags = 0;
93*de1e4e89SAndroid Build Coastguard Worker 	__u16 oflags = 0;
94*de1e4e89SAndroid Build Coastguard Worker 	unsigned int ikey = 0;
95*de1e4e89SAndroid Build Coastguard Worker 	unsigned int okey = 0;
96*de1e4e89SAndroid Build Coastguard Worker 	struct in6_addr raddr = IN6ADDR_ANY_INIT;
97*de1e4e89SAndroid Build Coastguard Worker 	struct in6_addr laddr = IN6ADDR_ANY_INIT;
98*de1e4e89SAndroid Build Coastguard Worker 	unsigned int link = 0;
99*de1e4e89SAndroid Build Coastguard Worker 	unsigned int flowinfo = 0;
100*de1e4e89SAndroid Build Coastguard Worker 	unsigned int flags = 0;
101*de1e4e89SAndroid Build Coastguard Worker 	__u8 hop_limit = DEFAULT_TNL_HOP_LIMIT;
102*de1e4e89SAndroid Build Coastguard Worker 	__u8 encap_limit = IPV6_DEFAULT_TNL_ENCAP_LIMIT;
103*de1e4e89SAndroid Build Coastguard Worker 	__u16 encaptype = 0;
104*de1e4e89SAndroid Build Coastguard Worker 	__u16 encapflags = TUNNEL_ENCAP_FLAG_CSUM6;
105*de1e4e89SAndroid Build Coastguard Worker 	__u16 encapsport = 0;
106*de1e4e89SAndroid Build Coastguard Worker 	__u16 encapdport = 0;
107*de1e4e89SAndroid Build Coastguard Worker 	int len;
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(greinfo, IFLA_GRE_MAX,
134*de1e4e89SAndroid Build Coastguard Worker 				    linkinfo[IFLA_INFO_DATA]);
135*de1e4e89SAndroid Build Coastguard Worker 
136*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_IKEY])
137*de1e4e89SAndroid Build Coastguard Worker 			ikey = rta_getattr_u32(greinfo[IFLA_GRE_IKEY]);
138*de1e4e89SAndroid Build Coastguard Worker 
139*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_OKEY])
140*de1e4e89SAndroid Build Coastguard Worker 			okey = rta_getattr_u32(greinfo[IFLA_GRE_OKEY]);
141*de1e4e89SAndroid Build Coastguard Worker 
142*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_IFLAGS])
143*de1e4e89SAndroid Build Coastguard Worker 			iflags = rta_getattr_u16(greinfo[IFLA_GRE_IFLAGS]);
144*de1e4e89SAndroid Build Coastguard Worker 
145*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_OFLAGS])
146*de1e4e89SAndroid Build Coastguard Worker 			oflags = rta_getattr_u16(greinfo[IFLA_GRE_OFLAGS]);
147*de1e4e89SAndroid Build Coastguard Worker 
148*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_LOCAL])
149*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&laddr, RTA_DATA(greinfo[IFLA_GRE_LOCAL]), sizeof(laddr));
150*de1e4e89SAndroid Build Coastguard Worker 
151*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_REMOTE])
152*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&raddr, RTA_DATA(greinfo[IFLA_GRE_REMOTE]), sizeof(raddr));
153*de1e4e89SAndroid Build Coastguard Worker 
154*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_TTL])
155*de1e4e89SAndroid Build Coastguard Worker 			hop_limit = rta_getattr_u8(greinfo[IFLA_GRE_TTL]);
156*de1e4e89SAndroid Build Coastguard Worker 
157*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_LINK])
158*de1e4e89SAndroid Build Coastguard Worker 			link = rta_getattr_u32(greinfo[IFLA_GRE_LINK]);
159*de1e4e89SAndroid Build Coastguard Worker 
160*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_ENCAP_LIMIT])
161*de1e4e89SAndroid Build Coastguard Worker 			encap_limit = rta_getattr_u8(greinfo[IFLA_GRE_ENCAP_LIMIT]);
162*de1e4e89SAndroid Build Coastguard Worker 
163*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_FLOWINFO])
164*de1e4e89SAndroid Build Coastguard Worker 			flowinfo = rta_getattr_u32(greinfo[IFLA_GRE_FLOWINFO]);
165*de1e4e89SAndroid Build Coastguard Worker 
166*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_FLAGS])
167*de1e4e89SAndroid Build Coastguard Worker 			flags = rta_getattr_u32(greinfo[IFLA_GRE_FLAGS]);
168*de1e4e89SAndroid Build Coastguard Worker 
169*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_ENCAP_TYPE])
170*de1e4e89SAndroid Build Coastguard Worker 			encaptype = rta_getattr_u16(greinfo[IFLA_GRE_ENCAP_TYPE]);
171*de1e4e89SAndroid Build Coastguard Worker 
172*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_ENCAP_FLAGS])
173*de1e4e89SAndroid Build Coastguard Worker 			encapflags = rta_getattr_u16(greinfo[IFLA_GRE_ENCAP_FLAGS]);
174*de1e4e89SAndroid Build Coastguard Worker 
175*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_ENCAP_SPORT])
176*de1e4e89SAndroid Build Coastguard Worker 			encapsport = rta_getattr_u16(greinfo[IFLA_GRE_ENCAP_SPORT]);
177*de1e4e89SAndroid Build Coastguard Worker 
178*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_ENCAP_DPORT])
179*de1e4e89SAndroid Build Coastguard Worker 			encapdport = rta_getattr_u16(greinfo[IFLA_GRE_ENCAP_DPORT]);
180*de1e4e89SAndroid Build Coastguard Worker 
181*de1e4e89SAndroid Build Coastguard Worker 		if (greinfo[IFLA_GRE_FWMARK])
182*de1e4e89SAndroid Build Coastguard Worker 			fwmark = rta_getattr_u32(greinfo[IFLA_GRE_FWMARK]);
183*de1e4e89SAndroid Build Coastguard Worker 	}
184*de1e4e89SAndroid Build Coastguard Worker 
185*de1e4e89SAndroid Build Coastguard Worker 	while (argc > 0) {
186*de1e4e89SAndroid Build Coastguard Worker 		if (!matches(*argv, "key")) {
187*de1e4e89SAndroid Build Coastguard Worker 			unsigned int uval;
188*de1e4e89SAndroid Build Coastguard Worker 
189*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
190*de1e4e89SAndroid Build Coastguard Worker 			iflags |= GRE_KEY;
191*de1e4e89SAndroid Build Coastguard Worker 			oflags |= GRE_KEY;
192*de1e4e89SAndroid Build Coastguard Worker 			if (strchr(*argv, '.'))
193*de1e4e89SAndroid Build Coastguard Worker 				uval = get_addr32(*argv);
194*de1e4e89SAndroid Build Coastguard Worker 			else {
195*de1e4e89SAndroid Build Coastguard Worker 				if (get_unsigned(&uval, *argv, 0) < 0) {
196*de1e4e89SAndroid Build Coastguard Worker 					fprintf(stderr,
197*de1e4e89SAndroid Build Coastguard Worker 						"Invalid value for \"key\"\n");
198*de1e4e89SAndroid Build Coastguard Worker 					exit(-1);
199*de1e4e89SAndroid Build Coastguard Worker 				}
200*de1e4e89SAndroid Build Coastguard Worker 				uval = htonl(uval);
201*de1e4e89SAndroid Build Coastguard Worker 			}
202*de1e4e89SAndroid Build Coastguard Worker 
203*de1e4e89SAndroid Build Coastguard Worker 			ikey = okey = uval;
204*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "ikey")) {
205*de1e4e89SAndroid Build Coastguard Worker 			unsigned int uval;
206*de1e4e89SAndroid Build Coastguard Worker 
207*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
208*de1e4e89SAndroid Build Coastguard Worker 			iflags |= GRE_KEY;
209*de1e4e89SAndroid Build Coastguard Worker 			if (strchr(*argv, '.'))
210*de1e4e89SAndroid Build Coastguard Worker 				uval = get_addr32(*argv);
211*de1e4e89SAndroid Build Coastguard Worker 			else {
212*de1e4e89SAndroid Build Coastguard Worker 				if (get_unsigned(&uval, *argv, 0) < 0) {
213*de1e4e89SAndroid Build Coastguard Worker 					fprintf(stderr, "invalid value of \"ikey\"\n");
214*de1e4e89SAndroid Build Coastguard Worker 					exit(-1);
215*de1e4e89SAndroid Build Coastguard Worker 				}
216*de1e4e89SAndroid Build Coastguard Worker 				uval = htonl(uval);
217*de1e4e89SAndroid Build Coastguard Worker 			}
218*de1e4e89SAndroid Build Coastguard Worker 			ikey = uval;
219*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "okey")) {
220*de1e4e89SAndroid Build Coastguard Worker 			unsigned int uval;
221*de1e4e89SAndroid Build Coastguard Worker 
222*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
223*de1e4e89SAndroid Build Coastguard Worker 			oflags |= GRE_KEY;
224*de1e4e89SAndroid Build Coastguard Worker 			if (strchr(*argv, '.'))
225*de1e4e89SAndroid Build Coastguard Worker 				uval = get_addr32(*argv);
226*de1e4e89SAndroid Build Coastguard Worker 			else {
227*de1e4e89SAndroid Build Coastguard Worker 				if (get_unsigned(&uval, *argv, 0) < 0) {
228*de1e4e89SAndroid Build Coastguard Worker 					fprintf(stderr, "invalid value of \"okey\"\n");
229*de1e4e89SAndroid Build Coastguard Worker 					exit(-1);
230*de1e4e89SAndroid Build Coastguard Worker 				}
231*de1e4e89SAndroid Build Coastguard Worker 				uval = htonl(uval);
232*de1e4e89SAndroid Build Coastguard Worker 			}
233*de1e4e89SAndroid Build Coastguard Worker 			okey = uval;
234*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "seq")) {
235*de1e4e89SAndroid Build Coastguard Worker 			iflags |= GRE_SEQ;
236*de1e4e89SAndroid Build Coastguard Worker 			oflags |= GRE_SEQ;
237*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "iseq")) {
238*de1e4e89SAndroid Build Coastguard Worker 			iflags |= GRE_SEQ;
239*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "oseq")) {
240*de1e4e89SAndroid Build Coastguard Worker 			oflags |= GRE_SEQ;
241*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "csum")) {
242*de1e4e89SAndroid Build Coastguard Worker 			iflags |= GRE_CSUM;
243*de1e4e89SAndroid Build Coastguard Worker 			oflags |= GRE_CSUM;
244*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "icsum")) {
245*de1e4e89SAndroid Build Coastguard Worker 			iflags |= GRE_CSUM;
246*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "ocsum")) {
247*de1e4e89SAndroid Build Coastguard Worker 			oflags |= GRE_CSUM;
248*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "remote")) {
249*de1e4e89SAndroid Build Coastguard Worker 			inet_prefix addr;
250*de1e4e89SAndroid Build Coastguard Worker 
251*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
252*de1e4e89SAndroid Build Coastguard Worker 			get_prefix(&addr, *argv, preferred_family);
253*de1e4e89SAndroid Build Coastguard Worker 			if (addr.family == AF_UNSPEC)
254*de1e4e89SAndroid Build Coastguard Worker 				invarg("\"remote\" address family is AF_UNSPEC", *argv);
255*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&raddr, &addr.data, sizeof(raddr));
256*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "local")) {
257*de1e4e89SAndroid Build Coastguard Worker 			inet_prefix addr;
258*de1e4e89SAndroid Build Coastguard Worker 
259*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
260*de1e4e89SAndroid Build Coastguard Worker 			get_prefix(&addr, *argv, preferred_family);
261*de1e4e89SAndroid Build Coastguard Worker 			if (addr.family == AF_UNSPEC)
262*de1e4e89SAndroid Build Coastguard Worker 				invarg("\"local\" address family is AF_UNSPEC", *argv);
263*de1e4e89SAndroid Build Coastguard Worker 			memcpy(&laddr, &addr.data, sizeof(laddr));
264*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "dev")) {
265*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
266*de1e4e89SAndroid Build Coastguard Worker 			link = if_nametoindex(*argv);
267*de1e4e89SAndroid Build Coastguard Worker 			if (link == 0) {
268*de1e4e89SAndroid Build Coastguard Worker 				fprintf(stderr, "Cannot find device \"%s\"\n",
269*de1e4e89SAndroid Build Coastguard Worker 					*argv);
270*de1e4e89SAndroid Build Coastguard Worker 				exit(-1);
271*de1e4e89SAndroid Build Coastguard Worker 			}
272*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "ttl") ||
273*de1e4e89SAndroid Build Coastguard Worker 			   !matches(*argv, "hoplimit")) {
274*de1e4e89SAndroid Build Coastguard Worker 			__u8 uval;
275*de1e4e89SAndroid Build Coastguard Worker 
276*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
277*de1e4e89SAndroid Build Coastguard Worker 			if (get_u8(&uval, *argv, 0))
278*de1e4e89SAndroid Build Coastguard Worker 				invarg("invalid TTL", *argv);
279*de1e4e89SAndroid Build Coastguard Worker 			hop_limit = uval;
280*de1e4e89SAndroid Build Coastguard Worker 		} else if (!matches(*argv, "tos") ||
281*de1e4e89SAndroid Build Coastguard Worker 			   !matches(*argv, "tclass") ||
282*de1e4e89SAndroid Build Coastguard Worker 			   !matches(*argv, "dsfield")) {
283*de1e4e89SAndroid Build Coastguard Worker 			__u8 uval;
284*de1e4e89SAndroid Build Coastguard Worker 
285*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
286*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "inherit") == 0)
287*de1e4e89SAndroid Build Coastguard Worker 				flags |= IP6_TNL_F_USE_ORIG_TCLASS;
288*de1e4e89SAndroid Build Coastguard Worker 			else {
289*de1e4e89SAndroid Build Coastguard Worker 				if (get_u8(&uval, *argv, 16))
290*de1e4e89SAndroid Build Coastguard Worker 					invarg("invalid TClass", *argv);
291*de1e4e89SAndroid Build Coastguard Worker 				flowinfo &= ~IP6_FLOWINFO_TCLASS;
292*de1e4e89SAndroid Build Coastguard Worker 				flowinfo |= htonl((__u32)uval << 20) & IP6_FLOWINFO_TCLASS;
293*de1e4e89SAndroid Build Coastguard Worker 				flags &= ~IP6_TNL_F_USE_ORIG_TCLASS;
294*de1e4e89SAndroid Build Coastguard Worker 			}
295*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "flowlabel") == 0 ||
296*de1e4e89SAndroid Build Coastguard Worker 			   strcmp(*argv, "fl") == 0) {
297*de1e4e89SAndroid Build Coastguard Worker 			__u32 uval;
298*de1e4e89SAndroid Build Coastguard Worker 
299*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
300*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "inherit") == 0)
301*de1e4e89SAndroid Build Coastguard Worker 				flags |= IP6_TNL_F_USE_ORIG_FLOWLABEL;
302*de1e4e89SAndroid Build Coastguard Worker 			else {
303*de1e4e89SAndroid Build Coastguard Worker 				if (get_u32(&uval, *argv, 16))
304*de1e4e89SAndroid Build Coastguard Worker 					invarg("invalid Flowlabel", *argv);
305*de1e4e89SAndroid Build Coastguard Worker 				if (uval > 0xFFFFF)
306*de1e4e89SAndroid Build Coastguard Worker 					invarg("invalid Flowlabel", *argv);
307*de1e4e89SAndroid Build Coastguard Worker 				flowinfo &= ~IP6_FLOWINFO_FLOWLABEL;
308*de1e4e89SAndroid Build Coastguard Worker 				flowinfo |= htonl(uval) & IP6_FLOWINFO_FLOWLABEL;
309*de1e4e89SAndroid Build Coastguard Worker 				flags &= ~IP6_TNL_F_USE_ORIG_FLOWLABEL;
310*de1e4e89SAndroid Build Coastguard Worker 			}
311*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "dscp") == 0) {
312*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
313*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "inherit") != 0)
314*de1e4e89SAndroid Build Coastguard Worker 				invarg("not inherit", *argv);
315*de1e4e89SAndroid Build Coastguard Worker 			flags |= IP6_TNL_F_RCV_DSCP_COPY;
316*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap") == 0) {
317*de1e4e89SAndroid Build Coastguard Worker 			encaptype = TUNNEL_ENCAP_NONE;
318*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap") == 0) {
319*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
320*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "fou") == 0)
321*de1e4e89SAndroid Build Coastguard Worker 				encaptype = TUNNEL_ENCAP_FOU;
322*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "gue") == 0)
323*de1e4e89SAndroid Build Coastguard Worker 				encaptype = TUNNEL_ENCAP_GUE;
324*de1e4e89SAndroid Build Coastguard Worker 			else if (strcmp(*argv, "none") == 0)
325*de1e4e89SAndroid Build Coastguard Worker 				encaptype = TUNNEL_ENCAP_NONE;
326*de1e4e89SAndroid Build Coastguard Worker 			else
327*de1e4e89SAndroid Build Coastguard Worker 				invarg("Invalid encap type.", *argv);
328*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-sport") == 0) {
329*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
330*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "auto") == 0)
331*de1e4e89SAndroid Build Coastguard Worker 				encapsport = 0;
332*de1e4e89SAndroid Build Coastguard Worker 			else if (get_u16(&encapsport, *argv, 0))
333*de1e4e89SAndroid Build Coastguard Worker 				invarg("Invalid source port.", *argv);
334*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-dport") == 0) {
335*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
336*de1e4e89SAndroid Build Coastguard Worker 			if (get_u16(&encapdport, *argv, 0))
337*de1e4e89SAndroid Build Coastguard Worker 				invarg("Invalid destination port.", *argv);
338*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-csum") == 0) {
339*de1e4e89SAndroid Build Coastguard Worker 			encapflags |= TUNNEL_ENCAP_FLAG_CSUM;
340*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap-csum") == 0) {
341*de1e4e89SAndroid Build Coastguard Worker 			encapflags &= ~TUNNEL_ENCAP_FLAG_CSUM;
342*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-udp6-csum") == 0) {
343*de1e4e89SAndroid Build Coastguard Worker 			encapflags |= TUNNEL_ENCAP_FLAG_CSUM6;
344*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap-udp6-csum") == 0) {
345*de1e4e89SAndroid Build Coastguard Worker 			encapflags &= ~TUNNEL_ENCAP_FLAG_CSUM6;
346*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encap-remcsum") == 0) {
347*de1e4e89SAndroid Build Coastguard Worker 			encapflags |= TUNNEL_ENCAP_FLAG_REMCSUM;
348*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "noencap-remcsum") == 0) {
349*de1e4e89SAndroid Build Coastguard Worker 			encapflags &= ~TUNNEL_ENCAP_FLAG_REMCSUM;
350*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "fwmark") == 0) {
351*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
352*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "inherit") == 0) {
353*de1e4e89SAndroid Build Coastguard Worker 				flags |= IP6_TNL_F_USE_ORIG_FWMARK;
354*de1e4e89SAndroid Build Coastguard Worker 				fwmark = 0;
355*de1e4e89SAndroid Build Coastguard Worker 			} else {
356*de1e4e89SAndroid Build Coastguard Worker 				if (get_u32(&fwmark, *argv, 0))
357*de1e4e89SAndroid Build Coastguard Worker 					invarg("invalid fwmark\n", *argv);
358*de1e4e89SAndroid Build Coastguard Worker 				flags &= ~IP6_TNL_F_USE_ORIG_FWMARK;
359*de1e4e89SAndroid Build Coastguard Worker 			}
360*de1e4e89SAndroid Build Coastguard Worker 		} else if (strcmp(*argv, "encaplimit") == 0) {
361*de1e4e89SAndroid Build Coastguard Worker 			NEXT_ARG();
362*de1e4e89SAndroid Build Coastguard Worker 			if (strcmp(*argv, "none") == 0) {
363*de1e4e89SAndroid Build Coastguard Worker 				flags |= IP6_TNL_F_IGN_ENCAP_LIMIT;
364*de1e4e89SAndroid Build Coastguard Worker 			} else {
365*de1e4e89SAndroid Build Coastguard Worker 				__u8 uval;
366*de1e4e89SAndroid Build Coastguard Worker 
367*de1e4e89SAndroid Build Coastguard Worker 				if (get_u8(&uval, *argv, 0) < -1)
368*de1e4e89SAndroid Build Coastguard Worker 					invarg("invalid ELIM", *argv);
369*de1e4e89SAndroid Build Coastguard Worker 				encap_limit = uval;
370*de1e4e89SAndroid Build Coastguard Worker 				flags &= ~IP6_TNL_F_IGN_ENCAP_LIMIT;
371*de1e4e89SAndroid Build Coastguard Worker 			}
372*de1e4e89SAndroid Build Coastguard Worker 		} else
373*de1e4e89SAndroid Build Coastguard Worker 			usage();
374*de1e4e89SAndroid Build Coastguard Worker 		argc--; argv++;
375*de1e4e89SAndroid Build Coastguard Worker 	}
376*de1e4e89SAndroid Build Coastguard Worker 
377*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_GRE_IKEY, ikey);
378*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_GRE_OKEY, okey);
379*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, 1024, IFLA_GRE_IFLAGS, &iflags, 2);
380*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, 1024, IFLA_GRE_OFLAGS, &oflags, 2);
381*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, 1024, IFLA_GRE_LOCAL, &laddr, sizeof(laddr));
382*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, 1024, IFLA_GRE_REMOTE, &raddr, sizeof(raddr));
383*de1e4e89SAndroid Build Coastguard Worker 	if (link)
384*de1e4e89SAndroid Build Coastguard Worker 		addattr32(n, 1024, IFLA_GRE_LINK, link);
385*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, 1024, IFLA_GRE_TTL, &hop_limit, 1);
386*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, 1024, IFLA_GRE_ENCAP_LIMIT, &encap_limit, 1);
387*de1e4e89SAndroid Build Coastguard Worker 	addattr_l(n, 1024, IFLA_GRE_FLOWINFO, &flowinfo, 4);
388*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_GRE_FLAGS, flags);
389*de1e4e89SAndroid Build Coastguard Worker 	addattr32(n, 1024, IFLA_GRE_FWMARK, fwmark);
390*de1e4e89SAndroid Build Coastguard Worker 
391*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_GRE_ENCAP_TYPE, encaptype);
392*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_GRE_ENCAP_FLAGS, encapflags);
393*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_GRE_ENCAP_SPORT, htons(encapsport));
394*de1e4e89SAndroid Build Coastguard Worker 	addattr16(n, 1024, IFLA_GRE_ENCAP_DPORT, htons(encapdport));
395*de1e4e89SAndroid Build Coastguard Worker 
396*de1e4e89SAndroid Build Coastguard Worker 	return 0;
397*de1e4e89SAndroid Build Coastguard Worker }
398*de1e4e89SAndroid Build Coastguard Worker 
gre_print_opt(struct link_util * lu,FILE * f,struct rtattr * tb[])399*de1e4e89SAndroid Build Coastguard Worker static void gre_print_opt(struct link_util *lu, FILE *f, struct rtattr *tb[])
400*de1e4e89SAndroid Build Coastguard Worker {
401*de1e4e89SAndroid Build Coastguard Worker 	char s2[64];
402*de1e4e89SAndroid Build Coastguard Worker 	const char *local = "any";
403*de1e4e89SAndroid Build Coastguard Worker 	const char *remote = "any";
404*de1e4e89SAndroid Build Coastguard Worker 	unsigned int iflags = 0;
405*de1e4e89SAndroid Build Coastguard Worker 	unsigned int oflags = 0;
406*de1e4e89SAndroid Build Coastguard Worker 	unsigned int flags = 0;
407*de1e4e89SAndroid Build Coastguard Worker 	unsigned int flowinfo = 0;
408*de1e4e89SAndroid Build Coastguard Worker 	struct in6_addr in6_addr_any = IN6ADDR_ANY_INIT;
409*de1e4e89SAndroid Build Coastguard Worker 
410*de1e4e89SAndroid Build Coastguard Worker 	if (!tb)
411*de1e4e89SAndroid Build Coastguard Worker 		return;
412*de1e4e89SAndroid Build Coastguard Worker 
413*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_FLAGS])
414*de1e4e89SAndroid Build Coastguard Worker 		flags = rta_getattr_u32(tb[IFLA_GRE_FLAGS]);
415*de1e4e89SAndroid Build Coastguard Worker 
416*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_FLOWINFO])
417*de1e4e89SAndroid Build Coastguard Worker 		flowinfo = rta_getattr_u32(tb[IFLA_GRE_FLOWINFO]);
418*de1e4e89SAndroid Build Coastguard Worker 
419*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_REMOTE]) {
420*de1e4e89SAndroid Build Coastguard Worker 		struct in6_addr addr;
421*de1e4e89SAndroid Build Coastguard Worker 
422*de1e4e89SAndroid Build Coastguard Worker 		memcpy(&addr, RTA_DATA(tb[IFLA_GRE_REMOTE]), sizeof(addr));
423*de1e4e89SAndroid Build Coastguard Worker 
424*de1e4e89SAndroid Build Coastguard Worker 		if (memcmp(&addr, &in6_addr_any, sizeof(addr)))
425*de1e4e89SAndroid Build Coastguard Worker 			remote = format_host(AF_INET6, sizeof(addr), &addr);
426*de1e4e89SAndroid Build Coastguard Worker 	}
427*de1e4e89SAndroid Build Coastguard Worker 
428*de1e4e89SAndroid Build Coastguard Worker 	print_string(PRINT_ANY, "remote", "remote %s ", remote);
429*de1e4e89SAndroid Build Coastguard Worker 
430*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_LOCAL]) {
431*de1e4e89SAndroid Build Coastguard Worker 		struct in6_addr addr;
432*de1e4e89SAndroid Build Coastguard Worker 
433*de1e4e89SAndroid Build Coastguard Worker 		memcpy(&addr, RTA_DATA(tb[IFLA_GRE_LOCAL]), sizeof(addr));
434*de1e4e89SAndroid Build Coastguard Worker 
435*de1e4e89SAndroid Build Coastguard Worker 		if (memcmp(&addr, &in6_addr_any, sizeof(addr)))
436*de1e4e89SAndroid Build Coastguard Worker 			local = format_host(AF_INET6, sizeof(addr), &addr);
437*de1e4e89SAndroid Build Coastguard Worker 	}
438*de1e4e89SAndroid Build Coastguard Worker 
439*de1e4e89SAndroid Build Coastguard Worker 	print_string(PRINT_ANY, "local", "local %s ", local);
440*de1e4e89SAndroid Build Coastguard Worker 
441*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_LINK] && rta_getattr_u32(tb[IFLA_GRE_LINK])) {
442*de1e4e89SAndroid Build Coastguard Worker 		unsigned int link = rta_getattr_u32(tb[IFLA_GRE_LINK]);
443*de1e4e89SAndroid Build Coastguard Worker 		const char *n = if_indextoname(link, s2);
444*de1e4e89SAndroid Build Coastguard Worker 
445*de1e4e89SAndroid Build Coastguard Worker 		if (n)
446*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "link", "dev %s ", n);
447*de1e4e89SAndroid Build Coastguard Worker 		else
448*de1e4e89SAndroid Build Coastguard Worker 			print_uint(PRINT_ANY, "link_index", "dev %u ", link);
449*de1e4e89SAndroid Build Coastguard Worker 	}
450*de1e4e89SAndroid Build Coastguard Worker 
451*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_TTL]) {
452*de1e4e89SAndroid Build Coastguard Worker 		__u8 ttl = rta_getattr_u8(tb[IFLA_GRE_TTL]);
453*de1e4e89SAndroid Build Coastguard Worker 
454*de1e4e89SAndroid Build Coastguard Worker 		if (ttl)
455*de1e4e89SAndroid Build Coastguard Worker 			print_int(PRINT_ANY, "ttl", "hoplimit %d ", ttl);
456*de1e4e89SAndroid Build Coastguard Worker 		else
457*de1e4e89SAndroid Build Coastguard Worker 			print_int(PRINT_JSON, "ttl", NULL, ttl);
458*de1e4e89SAndroid Build Coastguard Worker 	}
459*de1e4e89SAndroid Build Coastguard Worker 
460*de1e4e89SAndroid Build Coastguard Worker 	if (flags & IP6_TNL_F_IGN_ENCAP_LIMIT)
461*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY,
462*de1e4e89SAndroid Build Coastguard Worker 			   "ip6_tnl_f_ign_encap_limit",
463*de1e4e89SAndroid Build Coastguard Worker 			   "encaplimit none ",
464*de1e4e89SAndroid Build Coastguard Worker 			   true);
465*de1e4e89SAndroid Build Coastguard Worker 	else if (tb[IFLA_GRE_ENCAP_LIMIT]) {
466*de1e4e89SAndroid Build Coastguard Worker 		int encap_limit = rta_getattr_u8(tb[IFLA_GRE_ENCAP_LIMIT]);
467*de1e4e89SAndroid Build Coastguard Worker 
468*de1e4e89SAndroid Build Coastguard Worker 		print_int(PRINT_ANY,
469*de1e4e89SAndroid Build Coastguard Worker 			  "encap_limit",
470*de1e4e89SAndroid Build Coastguard Worker 			  "encaplimit %d ",
471*de1e4e89SAndroid Build Coastguard Worker 			  encap_limit);
472*de1e4e89SAndroid Build Coastguard Worker 	}
473*de1e4e89SAndroid Build Coastguard Worker 
474*de1e4e89SAndroid Build Coastguard Worker 	if (flags & IP6_TNL_F_USE_ORIG_FLOWLABEL) {
475*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY,
476*de1e4e89SAndroid Build Coastguard Worker 			   "ip6_tnl_f_use_orig_flowlabel",
477*de1e4e89SAndroid Build Coastguard Worker 			   "flowlabel inherit ",
478*de1e4e89SAndroid Build Coastguard Worker 			   true);
479*de1e4e89SAndroid Build Coastguard Worker 	} else {
480*de1e4e89SAndroid Build Coastguard Worker 		if (is_json_context()) {
481*de1e4e89SAndroid Build Coastguard Worker 			SPRINT_BUF(b1);
482*de1e4e89SAndroid Build Coastguard Worker 
483*de1e4e89SAndroid Build Coastguard Worker 			snprintf(b1, sizeof(b1), "0x%05x",
484*de1e4e89SAndroid Build Coastguard Worker 				 ntohl(flowinfo & IP6_FLOWINFO_FLOWLABEL));
485*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_JSON, "flowlabel", NULL, b1);
486*de1e4e89SAndroid Build Coastguard Worker 
487*de1e4e89SAndroid Build Coastguard Worker 		} else {
488*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "flowlabel 0x%05x ",
489*de1e4e89SAndroid Build Coastguard Worker 				ntohl(flowinfo & IP6_FLOWINFO_FLOWLABEL));
490*de1e4e89SAndroid Build Coastguard Worker 		}
491*de1e4e89SAndroid Build Coastguard Worker 	}
492*de1e4e89SAndroid Build Coastguard Worker 
493*de1e4e89SAndroid Build Coastguard Worker 	if (flags & IP6_TNL_F_USE_ORIG_TCLASS) {
494*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY,
495*de1e4e89SAndroid Build Coastguard Worker 			   "ip6_tnl_f_use_orig_tclass",
496*de1e4e89SAndroid Build Coastguard Worker 			   "tclass inherit ",
497*de1e4e89SAndroid Build Coastguard Worker 			   true);
498*de1e4e89SAndroid Build Coastguard Worker 	} else {
499*de1e4e89SAndroid Build Coastguard Worker 		if (is_json_context()) {
500*de1e4e89SAndroid Build Coastguard Worker 			SPRINT_BUF(b1);
501*de1e4e89SAndroid Build Coastguard Worker 
502*de1e4e89SAndroid Build Coastguard Worker 			snprintf(b1, sizeof(b1), "0x%05x",
503*de1e4e89SAndroid Build Coastguard Worker 				 ntohl(flowinfo & IP6_FLOWINFO_TCLASS) >> 20);
504*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_JSON, "tclass", NULL, b1);
505*de1e4e89SAndroid Build Coastguard Worker 		} else {
506*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "tclass 0x%02x ",
507*de1e4e89SAndroid Build Coastguard Worker 				 ntohl(flowinfo & IP6_FLOWINFO_TCLASS) >> 20);
508*de1e4e89SAndroid Build Coastguard Worker 		}
509*de1e4e89SAndroid Build Coastguard Worker 	}
510*de1e4e89SAndroid Build Coastguard Worker 
511*de1e4e89SAndroid Build Coastguard Worker 	if (flags & IP6_TNL_F_RCV_DSCP_COPY)
512*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY,
513*de1e4e89SAndroid Build Coastguard Worker 			   "ip6_tnl_f_rcv_dscp_copy",
514*de1e4e89SAndroid Build Coastguard Worker 			   "dscp inherit ",
515*de1e4e89SAndroid Build Coastguard Worker 			   true);
516*de1e4e89SAndroid Build Coastguard Worker 
517*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_IFLAGS])
518*de1e4e89SAndroid Build Coastguard Worker 		iflags = rta_getattr_u16(tb[IFLA_GRE_IFLAGS]);
519*de1e4e89SAndroid Build Coastguard Worker 
520*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_OFLAGS])
521*de1e4e89SAndroid Build Coastguard Worker 		oflags = rta_getattr_u16(tb[IFLA_GRE_OFLAGS]);
522*de1e4e89SAndroid Build Coastguard Worker 
523*de1e4e89SAndroid Build Coastguard Worker 	if ((iflags & GRE_KEY) && tb[IFLA_GRE_IKEY]) {
524*de1e4e89SAndroid Build Coastguard Worker 		inet_ntop(AF_INET, RTA_DATA(tb[IFLA_GRE_IKEY]), s2, sizeof(s2));
525*de1e4e89SAndroid Build Coastguard Worker 		print_string(PRINT_ANY, "ikey", "ikey %s ", s2);
526*de1e4e89SAndroid Build Coastguard Worker 	}
527*de1e4e89SAndroid Build Coastguard Worker 
528*de1e4e89SAndroid Build Coastguard Worker 	if ((oflags & GRE_KEY) && tb[IFLA_GRE_OKEY]) {
529*de1e4e89SAndroid Build Coastguard Worker 		inet_ntop(AF_INET, RTA_DATA(tb[IFLA_GRE_OKEY]), s2, sizeof(s2));
530*de1e4e89SAndroid Build Coastguard Worker 		print_string(PRINT_ANY, "okey", "okey %s ", s2);
531*de1e4e89SAndroid Build Coastguard Worker 	}
532*de1e4e89SAndroid Build Coastguard Worker 
533*de1e4e89SAndroid Build Coastguard Worker 	if (iflags & GRE_SEQ)
534*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY, "iseq", "iseq ", true);
535*de1e4e89SAndroid Build Coastguard Worker 	if (oflags & GRE_SEQ)
536*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY, "oseq", "oseq ", true);
537*de1e4e89SAndroid Build Coastguard Worker 	if (iflags & GRE_CSUM)
538*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY, "icsum", "icsum ", true);
539*de1e4e89SAndroid Build Coastguard Worker 	if (oflags & GRE_CSUM)
540*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY, "ocsum", "ocsum ", true);
541*de1e4e89SAndroid Build Coastguard Worker 
542*de1e4e89SAndroid Build Coastguard Worker 	if (flags & IP6_TNL_F_USE_ORIG_FWMARK)
543*de1e4e89SAndroid Build Coastguard Worker 		print_bool(PRINT_ANY,
544*de1e4e89SAndroid Build Coastguard Worker 			   "ip6_tnl_f_use_orig_fwmark",
545*de1e4e89SAndroid Build Coastguard Worker 			   "fwmark inherit ",
546*de1e4e89SAndroid Build Coastguard Worker 			   true);
547*de1e4e89SAndroid Build Coastguard Worker 	else if (tb[IFLA_GRE_FWMARK]) {
548*de1e4e89SAndroid Build Coastguard Worker 		__u32 fwmark = rta_getattr_u32(tb[IFLA_GRE_FWMARK]);
549*de1e4e89SAndroid Build Coastguard Worker 
550*de1e4e89SAndroid Build Coastguard Worker 		if (fwmark) {
551*de1e4e89SAndroid Build Coastguard Worker 			snprintf(s2, sizeof(s2), "0x%x", fwmark);
552*de1e4e89SAndroid Build Coastguard Worker 
553*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "fwmark", "fwmark %s ", s2);
554*de1e4e89SAndroid Build Coastguard Worker 		}
555*de1e4e89SAndroid Build Coastguard Worker 	}
556*de1e4e89SAndroid Build Coastguard Worker 
557*de1e4e89SAndroid Build Coastguard Worker 	if (tb[IFLA_GRE_ENCAP_TYPE] &&
558*de1e4e89SAndroid Build Coastguard Worker 	    rta_getattr_u16(tb[IFLA_GRE_ENCAP_TYPE]) != TUNNEL_ENCAP_NONE) {
559*de1e4e89SAndroid Build Coastguard Worker 		__u16 type = rta_getattr_u16(tb[IFLA_GRE_ENCAP_TYPE]);
560*de1e4e89SAndroid Build Coastguard Worker 		__u16 flags = rta_getattr_u16(tb[IFLA_GRE_ENCAP_FLAGS]);
561*de1e4e89SAndroid Build Coastguard Worker 		__u16 sport = rta_getattr_u16(tb[IFLA_GRE_ENCAP_SPORT]);
562*de1e4e89SAndroid Build Coastguard Worker 		__u16 dport = rta_getattr_u16(tb[IFLA_GRE_ENCAP_DPORT]);
563*de1e4e89SAndroid Build Coastguard Worker 
564*de1e4e89SAndroid Build Coastguard Worker 		open_json_object("encap");
565*de1e4e89SAndroid Build Coastguard Worker 
566*de1e4e89SAndroid Build Coastguard Worker 		print_string(PRINT_FP, NULL, "encap ", NULL);
567*de1e4e89SAndroid Build Coastguard Worker 		switch (type) {
568*de1e4e89SAndroid Build Coastguard Worker 		case TUNNEL_ENCAP_FOU:
569*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "type", "%s ", "fou");
570*de1e4e89SAndroid Build Coastguard Worker 			break;
571*de1e4e89SAndroid Build Coastguard Worker 		case TUNNEL_ENCAP_GUE:
572*de1e4e89SAndroid Build Coastguard Worker 			print_string(PRINT_ANY, "type", "%s ", "gue");
573*de1e4e89SAndroid Build Coastguard Worker 			break;
574*de1e4e89SAndroid Build Coastguard Worker 		default:
575*de1e4e89SAndroid Build Coastguard Worker 			print_null(PRINT_ANY, "type", "unknown ", NULL);
576*de1e4e89SAndroid Build Coastguard Worker 			break;
577*de1e4e89SAndroid Build Coastguard Worker 		}
578*de1e4e89SAndroid Build Coastguard Worker 
579*de1e4e89SAndroid Build Coastguard Worker 		if (is_json_context()) {
580*de1e4e89SAndroid Build Coastguard Worker 			print_uint(PRINT_JSON,
581*de1e4e89SAndroid Build Coastguard Worker 				   "sport",
582*de1e4e89SAndroid Build Coastguard Worker 				   NULL,
583*de1e4e89SAndroid Build Coastguard Worker 				   sport ? ntohs(sport) : 0);
584*de1e4e89SAndroid Build Coastguard Worker 			print_uint(PRINT_JSON, "dport", NULL, ntohs(dport));
585*de1e4e89SAndroid Build Coastguard Worker 			print_bool(PRINT_JSON, "csum", NULL,
586*de1e4e89SAndroid Build Coastguard Worker 					   flags & TUNNEL_ENCAP_FLAG_CSUM);
587*de1e4e89SAndroid Build Coastguard Worker 			print_bool(PRINT_JSON, "csum6", NULL,
588*de1e4e89SAndroid Build Coastguard Worker 					   flags & TUNNEL_ENCAP_FLAG_CSUM6);
589*de1e4e89SAndroid Build Coastguard Worker 			print_bool(PRINT_JSON, "remcsum", NULL,
590*de1e4e89SAndroid Build Coastguard Worker 					   flags & TUNNEL_ENCAP_FLAG_REMCSUM);
591*de1e4e89SAndroid Build Coastguard Worker 			close_json_object();
592*de1e4e89SAndroid Build Coastguard Worker 		} else {
593*de1e4e89SAndroid Build Coastguard Worker 			if (sport == 0)
594*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-sport auto ", f);
595*de1e4e89SAndroid Build Coastguard Worker 			else
596*de1e4e89SAndroid Build Coastguard Worker 				fprintf(f, "encap-sport %u", ntohs(sport));
597*de1e4e89SAndroid Build Coastguard Worker 
598*de1e4e89SAndroid Build Coastguard Worker 			fprintf(f, "encap-dport %u ", ntohs(dport));
599*de1e4e89SAndroid Build Coastguard Worker 
600*de1e4e89SAndroid Build Coastguard Worker 			if (flags & TUNNEL_ENCAP_FLAG_CSUM)
601*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-csum ", f);
602*de1e4e89SAndroid Build Coastguard Worker 			else
603*de1e4e89SAndroid Build Coastguard Worker 				fputs("noencap-csum ", f);
604*de1e4e89SAndroid Build Coastguard Worker 
605*de1e4e89SAndroid Build Coastguard Worker 			if (flags & TUNNEL_ENCAP_FLAG_CSUM6)
606*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-csum6 ", f);
607*de1e4e89SAndroid Build Coastguard Worker 			else
608*de1e4e89SAndroid Build Coastguard Worker 				fputs("noencap-csum6 ", f);
609*de1e4e89SAndroid Build Coastguard Worker 
610*de1e4e89SAndroid Build Coastguard Worker 			if (flags & TUNNEL_ENCAP_FLAG_REMCSUM)
611*de1e4e89SAndroid Build Coastguard Worker 				fputs("encap-remcsum ", f);
612*de1e4e89SAndroid Build Coastguard Worker 			else
613*de1e4e89SAndroid Build Coastguard Worker 				fputs("noencap-remcsum ", f);
614*de1e4e89SAndroid Build Coastguard Worker 		}
615*de1e4e89SAndroid Build Coastguard Worker 	}
616*de1e4e89SAndroid Build Coastguard Worker }
617*de1e4e89SAndroid Build Coastguard Worker 
gre_print_help(struct link_util * lu,int argc,char ** argv,FILE * f)618*de1e4e89SAndroid Build Coastguard Worker static void gre_print_help(struct link_util *lu, int argc, char **argv,
619*de1e4e89SAndroid Build Coastguard Worker 	FILE *f)
620*de1e4e89SAndroid Build Coastguard Worker {
621*de1e4e89SAndroid Build Coastguard Worker 	print_usage(f);
622*de1e4e89SAndroid Build Coastguard Worker }
623*de1e4e89SAndroid Build Coastguard Worker 
624*de1e4e89SAndroid Build Coastguard Worker struct link_util ip6gre_link_util = {
625*de1e4e89SAndroid Build Coastguard Worker 	.id = "ip6gre",
626*de1e4e89SAndroid Build Coastguard Worker 	.maxattr = IFLA_GRE_MAX,
627*de1e4e89SAndroid Build Coastguard Worker 	.parse_opt = gre_parse_opt,
628*de1e4e89SAndroid Build Coastguard Worker 	.print_opt = gre_print_opt,
629*de1e4e89SAndroid Build Coastguard Worker 	.print_help = gre_print_help,
630*de1e4e89SAndroid Build Coastguard Worker };
631*de1e4e89SAndroid Build Coastguard Worker 
632*de1e4e89SAndroid Build Coastguard Worker struct link_util ip6gretap_link_util = {
633*de1e4e89SAndroid Build Coastguard Worker 	.id = "ip6gretap",
634*de1e4e89SAndroid Build Coastguard Worker 	.maxattr = IFLA_GRE_MAX,
635*de1e4e89SAndroid Build Coastguard Worker 	.parse_opt = gre_parse_opt,
636*de1e4e89SAndroid Build Coastguard Worker 	.print_opt = gre_print_opt,
637*de1e4e89SAndroid Build Coastguard Worker 	.print_help = gre_print_help,
638*de1e4e89SAndroid Build Coastguard Worker };
639