2 * iplink_vxlan.c VXLAN device support
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version
7 * 2 of the License, or (at your option) any later version.
9 * Authors: Stephen Hemminger <shemminger@vyatta.com
17 #include <linux/if_link.h>
18 #include <arpa/inet.h>
22 #include "ip_common.h"
24 static void print_explain(FILE *f
)
27 "Usage: ... vxlan id VNI\n"
28 " [ { group | remote } IP_ADDRESS ]\n"
32 " [ flowlabel LABEL ]\n"
35 " [ srcport MIN MAX ]\n"
41 " [ ageing SECONDS ]\n"
42 " [ maxaddress NUMBER ]\n"
44 " [ [no]udp6zerocsumtx ]\n"
45 " [ [no]udp6zerocsumrx ]\n"
46 " [ [no]remcsumtx ] [ [no]remcsumrx ]\n"
47 " [ [no]external ] [ gbp ] [ gpe ]\n"
49 "Where: VNI := 0-16777215\n"
50 " ADDR := { IP_ADDRESS | any }\n"
51 " TOS := { NUMBER | inherit }\n"
52 " TTL := { 1..255 | inherit }\n"
53 " LABEL := 0-1048575\n"
57 static void explain(void)
59 print_explain(stderr
);
62 static int vxlan_parse_opt(struct link_util
*lu
, int argc
, char **argv
,
70 struct in6_addr saddr6
= IN6ADDR_ANY_INIT
;
71 struct in6_addr gaddr6
= IN6ADDR_ANY_INIT
;
72 struct in6_addr daddr6
= IN6ADDR_ANY_INIT
;
73 unsigned int link
= 0;
87 bool udpcsum_set
= false;
88 __u8 udp6zerocsumtx
= 0;
89 bool udp6zerocsumtx_set
= false;
90 __u8 udp6zerocsumrx
= 0;
91 bool udp6zerocsumrx_set
= false;
98 struct ifla_vxlan_port_range range
= { 0, 0 };
101 if (!matches(*argv
, "id") ||
102 !matches(*argv
, "vni")) {
104 if (get_u32(&vni
, *argv
, 0) ||
106 invarg("invalid id", *argv
);
108 } else if (!matches(*argv
, "group")) {
110 if (!inet_get_addr(*argv
, &gaddr
, &gaddr6
)) {
111 fprintf(stderr
, "Invalid address \"%s\"\n", *argv
);
114 if (!IN6_IS_ADDR_MULTICAST(&gaddr6
) && !IN_MULTICAST(ntohl(gaddr
)))
115 invarg("invalid group address", *argv
);
116 } else if (!matches(*argv
, "remote")) {
118 if (!inet_get_addr(*argv
, &daddr
, &daddr6
)) {
119 fprintf(stderr
, "Invalid address \"%s\"\n", *argv
);
122 if (IN6_IS_ADDR_MULTICAST(&daddr6
) || IN_MULTICAST(ntohl(daddr
)))
123 invarg("invalid remote address", *argv
);
124 } else if (!matches(*argv
, "local")) {
126 if (strcmp(*argv
, "any")) {
127 if (!inet_get_addr(*argv
, &saddr
, &saddr6
)) {
128 fprintf(stderr
, "Invalid address \"%s\"\n", *argv
);
133 if (IN_MULTICAST(ntohl(saddr
)) || IN6_IS_ADDR_MULTICAST(&saddr6
))
134 invarg("invalid local address", *argv
);
135 } else if (!matches(*argv
, "dev")) {
137 link
= if_nametoindex(*argv
);
139 fprintf(stderr
, "Cannot find device \"%s\"\n",
143 } else if (!matches(*argv
, "ttl") ||
144 !matches(*argv
, "hoplimit")) {
148 if (strcmp(*argv
, "inherit") != 0) {
149 if (get_unsigned(&uval
, *argv
, 0))
150 invarg("invalid TTL", *argv
);
152 invarg("TTL must be <= 255", *argv
);
155 } else if (!matches(*argv
, "tos") ||
156 !matches(*argv
, "dsfield")) {
160 if (strcmp(*argv
, "inherit") != 0) {
161 if (rtnl_dsfield_a2n(&uval
, *argv
))
162 invarg("bad TOS value", *argv
);
166 } else if (!matches(*argv
, "label") ||
167 !matches(*argv
, "flowlabel")) {
171 if (get_u32(&uval
, *argv
, 0) ||
172 (uval
& ~LABEL_MAX_MASK
))
173 invarg("invalid flowlabel", *argv
);
175 } else if (!matches(*argv
, "ageing")) {
177 if (strcmp(*argv
, "none") == 0)
179 else if (get_u32(&age
, *argv
, 0))
180 invarg("ageing timer", *argv
);
181 } else if (!matches(*argv
, "maxaddress")) {
183 if (strcmp(*argv
, "unlimited") == 0)
185 else if (get_u32(&maxaddr
, *argv
, 0))
186 invarg("max addresses", *argv
);
187 } else if (!matches(*argv
, "port") ||
188 !matches(*argv
, "srcport")) {
190 if (get_be16(&range
.low
, *argv
, 0))
191 invarg("min port", *argv
);
193 if (get_be16(&range
.high
, *argv
, 0))
194 invarg("max port", *argv
);
195 } else if (!matches(*argv
, "dstport")) {
197 if (get_u16(&dstport
, *argv
, 0))
198 invarg("dst port", *argv
);
200 } else if (!matches(*argv
, "nolearning")) {
202 } else if (!matches(*argv
, "learning")) {
204 } else if (!matches(*argv
, "noproxy")) {
206 } else if (!matches(*argv
, "proxy")) {
208 } else if (!matches(*argv
, "norsc")) {
210 } else if (!matches(*argv
, "rsc")) {
212 } else if (!matches(*argv
, "nol2miss")) {
214 } else if (!matches(*argv
, "l2miss")) {
216 } else if (!matches(*argv
, "nol3miss")) {
218 } else if (!matches(*argv
, "l3miss")) {
220 } else if (!matches(*argv
, "udpcsum")) {
223 } else if (!matches(*argv
, "noudpcsum")) {
226 } else if (!matches(*argv
, "udp6zerocsumtx")) {
228 udp6zerocsumtx_set
= true;
229 } else if (!matches(*argv
, "noudp6zerocsumtx")) {
231 udp6zerocsumtx_set
= true;
232 } else if (!matches(*argv
, "udp6zerocsumrx")) {
234 udp6zerocsumrx_set
= true;
235 } else if (!matches(*argv
, "noudp6zerocsumrx")) {
237 udp6zerocsumrx_set
= true;
238 } else if (!matches(*argv
, "remcsumtx")) {
240 } else if (!matches(*argv
, "noremcsumtx")) {
242 } else if (!matches(*argv
, "remcsumrx")) {
244 } else if (!matches(*argv
, "noremcsumrx")) {
246 } else if (!matches(*argv
, "external")) {
249 } else if (!matches(*argv
, "noexternal")) {
251 } else if (!matches(*argv
, "gbp")) {
253 } else if (!matches(*argv
, "gpe")) {
255 } else if (matches(*argv
, "help") == 0) {
259 fprintf(stderr
, "vxlan: unknown command \"%s\"?\n", *argv
);
266 if (metadata
&& vni_set
) {
267 fprintf(stderr
, "vxlan: both 'external' and vni cannot be specified\n");
271 if (!metadata
&& !vni_set
) {
272 fprintf(stderr
, "vxlan: missing virtual network identifier\n");
276 if ((gaddr
&& daddr
) ||
277 (!IN6_IS_ADDR_UNSPECIFIED(&gaddr6
) &&
278 !IN6_IS_ADDR_UNSPECIFIED(&daddr6
))) {
279 fprintf(stderr
, "vxlan: both group and remote cannot be specified\n");
283 if ((gaddr
|| !IN6_IS_ADDR_UNSPECIFIED(&gaddr6
)) && !link
) {
284 fprintf(stderr
, "vxlan: 'group' requires 'dev' to be specified\n");
288 if (!dst_port_set
&& gpe
) {
290 } else if (!dst_port_set
) {
291 fprintf(stderr
, "vxlan: destination port not specified\n"
292 "Will use Linux kernel default (non-standard value)\n");
294 "Use 'dstport 4789' to get the IANA assigned value\n"
295 "Use 'dstport 0' to get default and quiet this message\n");
298 addattr32(n
, 1024, IFLA_VXLAN_ID
, vni
);
300 addattr_l(n
, 1024, IFLA_VXLAN_GROUP
, &gaddr
, 4);
302 addattr_l(n
, 1024, IFLA_VXLAN_GROUP
, &daddr
, 4);
303 else if (!IN6_IS_ADDR_UNSPECIFIED(&gaddr6
))
304 addattr_l(n
, 1024, IFLA_VXLAN_GROUP6
, &gaddr6
, sizeof(struct in6_addr
));
305 else if (!IN6_IS_ADDR_UNSPECIFIED(&daddr6
))
306 addattr_l(n
, 1024, IFLA_VXLAN_GROUP6
, &daddr6
, sizeof(struct in6_addr
));
307 else if (preferred_family
== AF_INET
)
308 addattr_l(n
, 1024, IFLA_VXLAN_GROUP
, &daddr
, 4);
309 else if (preferred_family
== AF_INET6
)
310 addattr_l(n
, 1024, IFLA_VXLAN_GROUP6
, &daddr6
, sizeof(struct in6_addr
));
313 addattr_l(n
, 1024, IFLA_VXLAN_LOCAL
, &saddr
, 4);
314 else if (!IN6_IS_ADDR_UNSPECIFIED(&saddr6
))
315 addattr_l(n
, 1024, IFLA_VXLAN_LOCAL6
, &saddr6
, sizeof(struct in6_addr
));
318 addattr32(n
, 1024, IFLA_VXLAN_LINK
, link
);
319 addattr32(n
, 1024, IFLA_VXLAN_LABEL
, label
);
320 addattr8(n
, 1024, IFLA_VXLAN_TTL
, ttl
);
321 addattr8(n
, 1024, IFLA_VXLAN_TOS
, tos
);
322 addattr8(n
, 1024, IFLA_VXLAN_LEARNING
, learning
);
323 addattr8(n
, 1024, IFLA_VXLAN_PROXY
, proxy
);
324 addattr8(n
, 1024, IFLA_VXLAN_RSC
, rsc
);
325 addattr8(n
, 1024, IFLA_VXLAN_L2MISS
, l2miss
);
326 addattr8(n
, 1024, IFLA_VXLAN_L3MISS
, l3miss
);
327 addattr8(n
, 1024, IFLA_VXLAN_REMCSUM_TX
, remcsumtx
);
328 addattr8(n
, 1024, IFLA_VXLAN_REMCSUM_RX
, remcsumrx
);
329 addattr8(n
, 1024, IFLA_VXLAN_COLLECT_METADATA
, metadata
);
332 addattr8(n
, 1024, IFLA_VXLAN_UDP_CSUM
, udpcsum
);
333 if (udp6zerocsumtx_set
)
334 addattr8(n
, 1024, IFLA_VXLAN_UDP_ZERO_CSUM6_TX
, udp6zerocsumtx
);
335 if (udp6zerocsumrx_set
)
336 addattr8(n
, 1024, IFLA_VXLAN_UDP_ZERO_CSUM6_RX
, udp6zerocsumrx
);
338 addattr32(n
, 1024, IFLA_VXLAN_AGEING
, 0);
340 addattr32(n
, 1024, IFLA_VXLAN_AGEING
, age
);
342 addattr32(n
, 1024, IFLA_VXLAN_LIMIT
, maxaddr
);
343 if (range
.low
|| range
.high
)
344 addattr_l(n
, 1024, IFLA_VXLAN_PORT_RANGE
,
345 &range
, sizeof(range
));
347 addattr16(n
, 1024, IFLA_VXLAN_PORT
, htons(dstport
));
350 addattr_l(n
, 1024, IFLA_VXLAN_GBP
, NULL
, 0);
352 addattr_l(n
, 1024, IFLA_VXLAN_GPE
, NULL
, 0);
358 static void vxlan_print_opt(struct link_util
*lu
, FILE *f
, struct rtattr
*tb
[])
369 if (!tb
[IFLA_VXLAN_ID
] ||
370 RTA_PAYLOAD(tb
[IFLA_VXLAN_ID
]) < sizeof(__u32
))
373 vni
= rta_getattr_u32(tb
[IFLA_VXLAN_ID
]);
374 fprintf(f
, "id %u ", vni
);
376 if (tb
[IFLA_VXLAN_GROUP
]) {
377 __be32 addr
= rta_getattr_u32(tb
[IFLA_VXLAN_GROUP
]);
380 if (IN_MULTICAST(ntohl(addr
)))
381 fprintf(f
, "group %s ",
382 format_host(AF_INET
, 4, &addr
));
384 fprintf(f
, "remote %s ",
385 format_host(AF_INET
, 4, &addr
));
387 } else if (tb
[IFLA_VXLAN_GROUP6
]) {
388 struct in6_addr addr
;
390 memcpy(&addr
, RTA_DATA(tb
[IFLA_VXLAN_GROUP6
]), sizeof(struct in6_addr
));
391 if (!IN6_IS_ADDR_UNSPECIFIED(&addr
)) {
392 if (IN6_IS_ADDR_MULTICAST(&addr
))
393 fprintf(f
, "group %s ",
394 format_host(AF_INET6
, sizeof(struct in6_addr
), &addr
));
396 fprintf(f
, "remote %s ",
397 format_host(AF_INET6
, sizeof(struct in6_addr
), &addr
));
401 if (tb
[IFLA_VXLAN_LOCAL
]) {
402 __be32 addr
= rta_getattr_u32(tb
[IFLA_VXLAN_LOCAL
]);
405 fprintf(f
, "local %s ",
406 format_host(AF_INET
, 4, &addr
));
407 } else if (tb
[IFLA_VXLAN_LOCAL6
]) {
408 struct in6_addr addr
;
410 memcpy(&addr
, RTA_DATA(tb
[IFLA_VXLAN_LOCAL6
]), sizeof(struct in6_addr
));
411 if (!IN6_IS_ADDR_UNSPECIFIED(&addr
))
412 fprintf(f
, "local %s ",
413 format_host(AF_INET6
, sizeof(struct in6_addr
), &addr
));
416 if (tb
[IFLA_VXLAN_LINK
] &&
417 (link
= rta_getattr_u32(tb
[IFLA_VXLAN_LINK
]))) {
418 const char *n
= if_indextoname(link
, s2
);
421 fprintf(f
, "dev %s ", n
);
423 fprintf(f
, "dev %u ", link
);
426 if (tb
[IFLA_VXLAN_PORT_RANGE
]) {
427 const struct ifla_vxlan_port_range
*r
428 = RTA_DATA(tb
[IFLA_VXLAN_PORT_RANGE
]);
429 fprintf(f
, "srcport %u %u ", ntohs(r
->low
), ntohs(r
->high
));
432 if (tb
[IFLA_VXLAN_PORT
])
433 fprintf(f
, "dstport %u ",
434 rta_getattr_be16(tb
[IFLA_VXLAN_PORT
]));
436 if (tb
[IFLA_VXLAN_LEARNING
] &&
437 !rta_getattr_u8(tb
[IFLA_VXLAN_LEARNING
]))
438 fputs("nolearning ", f
);
440 if (tb
[IFLA_VXLAN_PROXY
] && rta_getattr_u8(tb
[IFLA_VXLAN_PROXY
]))
443 if (tb
[IFLA_VXLAN_RSC
] && rta_getattr_u8(tb
[IFLA_VXLAN_RSC
]))
446 if (tb
[IFLA_VXLAN_L2MISS
] && rta_getattr_u8(tb
[IFLA_VXLAN_L2MISS
]))
449 if (tb
[IFLA_VXLAN_L3MISS
] && rta_getattr_u8(tb
[IFLA_VXLAN_L3MISS
]))
452 if (tb
[IFLA_VXLAN_TOS
] &&
453 (tos
= rta_getattr_u8(tb
[IFLA_VXLAN_TOS
]))) {
455 fprintf(f
, "tos inherit ");
457 fprintf(f
, "tos %#x ", tos
);
460 if (tb
[IFLA_VXLAN_TTL
]) {
461 __u8 ttl
= rta_getattr_u8(tb
[IFLA_VXLAN_TTL
]);
464 fprintf(f
, "ttl %d ", ttl
);
467 if (tb
[IFLA_VXLAN_LABEL
]) {
468 __u32 label
= rta_getattr_u32(tb
[IFLA_VXLAN_LABEL
]);
471 fprintf(f
, "flowlabel %#x ", ntohl(label
));
474 if (tb
[IFLA_VXLAN_AGEING
]) {
475 __u32 age
= rta_getattr_u32(tb
[IFLA_VXLAN_AGEING
]);
478 fprintf(f
, "ageing none ");
480 fprintf(f
, "ageing %u ", age
);
483 if (tb
[IFLA_VXLAN_LIMIT
] &&
484 ((maxaddr
= rta_getattr_u32(tb
[IFLA_VXLAN_LIMIT
])) != 0))
485 fprintf(f
, "maxaddr %u ", maxaddr
);
487 if (tb
[IFLA_VXLAN_UDP_CSUM
]) {
488 if (!rta_getattr_u8(tb
[IFLA_VXLAN_UDP_CSUM
]))
490 fputs("udpcsum ", f
);
493 if (tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_TX
]) {
494 if (!rta_getattr_u8(tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_TX
]))
496 fputs("udp6zerocsumtx ", f
);
499 if (tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_RX
]) {
500 if (!rta_getattr_u8(tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_RX
]))
502 fputs("udp6zerocsumrx ", f
);
505 if (tb
[IFLA_VXLAN_REMCSUM_TX
] &&
506 rta_getattr_u8(tb
[IFLA_VXLAN_REMCSUM_TX
]))
507 fputs("remcsumtx ", f
);
509 if (tb
[IFLA_VXLAN_REMCSUM_RX
] &&
510 rta_getattr_u8(tb
[IFLA_VXLAN_REMCSUM_RX
]))
511 fputs("remcsumrx ", f
);
513 if (tb
[IFLA_VXLAN_COLLECT_METADATA
] &&
514 rta_getattr_u8(tb
[IFLA_VXLAN_COLLECT_METADATA
]))
515 fputs("external ", f
);
517 if (tb
[IFLA_VXLAN_GBP
])
519 if (tb
[IFLA_VXLAN_GPE
])
523 static void vxlan_print_help(struct link_util
*lu
, int argc
, char **argv
,
529 struct link_util vxlan_link_util
= {
531 .maxattr
= IFLA_VXLAN_MAX
,
532 .parse_opt
= vxlan_parse_opt
,
533 .print_opt
= vxlan_print_opt
,
534 .print_help
= vxlan_print_help
,