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
)
26 fprintf(f
, "Usage: ... vxlan id VNI [ { group | remote } IP_ADDRESS ] [ local ADDR ]\n");
27 fprintf(f
, " [ ttl TTL ] [ tos TOS ] [ flowlabel LABEL ] [ dev PHYS_DEV ]\n");
28 fprintf(f
, " [ dstport PORT ] [ srcport MIN MAX ]\n");
29 fprintf(f
, " [ [no]learning ] [ [no]proxy ] [ [no]rsc ]\n");
30 fprintf(f
, " [ [no]l2miss ] [ [no]l3miss ]\n");
31 fprintf(f
, " [ ageing SECONDS ] [ maxaddress NUMBER ]\n");
32 fprintf(f
, " [ [no]udpcsum ] [ [no]udp6zerocsumtx ] [ [no]udp6zerocsumrx ]\n");
33 fprintf(f
, " [ [no]remcsumtx ] [ [no]remcsumrx ]\n");
34 fprintf(f
, " [ [no]external ] [ gbp ] [ gpe ]\n");
36 fprintf(f
, "Where: VNI := 0-16777215\n");
37 fprintf(f
, " ADDR := { IP_ADDRESS | any }\n");
38 fprintf(f
, " TOS := { NUMBER | inherit }\n");
39 fprintf(f
, " TTL := { 1..255 | inherit }\n");
40 fprintf(f
, " LABEL := 0-1048575\n");
43 static void explain(void)
45 print_explain(stderr
);
48 static int vxlan_parse_opt(struct link_util
*lu
, int argc
, char **argv
,
56 struct in6_addr saddr6
= IN6ADDR_ANY_INIT
;
57 struct in6_addr gaddr6
= IN6ADDR_ANY_INIT
;
58 struct in6_addr daddr6
= IN6ADDR_ANY_INIT
;
59 unsigned int link
= 0;
73 bool udpcsum_set
= false;
74 __u8 udp6zerocsumtx
= 0;
75 bool udp6zerocsumtx_set
= false;
76 __u8 udp6zerocsumrx
= 0;
77 bool udp6zerocsumrx_set
= false;
84 struct ifla_vxlan_port_range range
= { 0, 0 };
87 if (!matches(*argv
, "id") ||
88 !matches(*argv
, "vni")) {
90 if (get_u32(&vni
, *argv
, 0) ||
92 invarg("invalid id", *argv
);
94 } else if (!matches(*argv
, "group")) {
96 if (!inet_get_addr(*argv
, &gaddr
, &gaddr6
)) {
97 fprintf(stderr
, "Invalid address \"%s\"\n", *argv
);
100 if (!IN6_IS_ADDR_MULTICAST(&gaddr6
) && !IN_MULTICAST(ntohl(gaddr
)))
101 invarg("invalid group address", *argv
);
102 } else if (!matches(*argv
, "remote")) {
104 if (!inet_get_addr(*argv
, &daddr
, &daddr6
)) {
105 fprintf(stderr
, "Invalid address \"%s\"\n", *argv
);
108 if (IN6_IS_ADDR_MULTICAST(&daddr6
) || IN_MULTICAST(ntohl(daddr
)))
109 invarg("invalid remote address", *argv
);
110 } else if (!matches(*argv
, "local")) {
112 if (strcmp(*argv
, "any")) {
113 if (!inet_get_addr(*argv
, &saddr
, &saddr6
)) {
114 fprintf(stderr
, "Invalid address \"%s\"\n", *argv
);
119 if (IN_MULTICAST(ntohl(saddr
)) || IN6_IS_ADDR_MULTICAST(&saddr6
))
120 invarg("invalid local address", *argv
);
121 } else if (!matches(*argv
, "dev")) {
123 link
= if_nametoindex(*argv
);
125 fprintf(stderr
, "Cannot find device \"%s\"\n",
129 } else if (!matches(*argv
, "ttl") ||
130 !matches(*argv
, "hoplimit")) {
134 if (strcmp(*argv
, "inherit") != 0) {
135 if (get_unsigned(&uval
, *argv
, 0))
136 invarg("invalid TTL", *argv
);
138 invarg("TTL must be <= 255", *argv
);
141 } else if (!matches(*argv
, "tos") ||
142 !matches(*argv
, "dsfield")) {
146 if (strcmp(*argv
, "inherit") != 0) {
147 if (rtnl_dsfield_a2n(&uval
, *argv
))
148 invarg("bad TOS value", *argv
);
152 } else if (!matches(*argv
, "label") ||
153 !matches(*argv
, "flowlabel")) {
157 if (get_u32(&uval
, *argv
, 0) ||
158 (uval
& ~LABEL_MAX_MASK
))
159 invarg("invalid flowlabel", *argv
);
161 } else if (!matches(*argv
, "ageing")) {
163 if (strcmp(*argv
, "none") == 0)
165 else if (get_u32(&age
, *argv
, 0))
166 invarg("ageing timer", *argv
);
167 } else if (!matches(*argv
, "maxaddress")) {
169 if (strcmp(*argv
, "unlimited") == 0)
171 else if (get_u32(&maxaddr
, *argv
, 0))
172 invarg("max addresses", *argv
);
173 } else if (!matches(*argv
, "port") ||
174 !matches(*argv
, "srcport")) {
175 __u16 minport
, maxport
;
178 if (get_u16(&minport
, *argv
, 0))
179 invarg("min port", *argv
);
181 if (get_u16(&maxport
, *argv
, 0))
182 invarg("max port", *argv
);
183 range
.low
= htons(minport
);
184 range
.high
= htons(maxport
);
185 } else if (!matches(*argv
, "dstport")) {
187 if (get_u16(&dstport
, *argv
, 0))
188 invarg("dst port", *argv
);
190 } else if (!matches(*argv
, "nolearning")) {
192 } else if (!matches(*argv
, "learning")) {
194 } else if (!matches(*argv
, "noproxy")) {
196 } else if (!matches(*argv
, "proxy")) {
198 } else if (!matches(*argv
, "norsc")) {
200 } else if (!matches(*argv
, "rsc")) {
202 } else if (!matches(*argv
, "nol2miss")) {
204 } else if (!matches(*argv
, "l2miss")) {
206 } else if (!matches(*argv
, "nol3miss")) {
208 } else if (!matches(*argv
, "l3miss")) {
210 } else if (!matches(*argv
, "udpcsum")) {
213 } else if (!matches(*argv
, "noudpcsum")) {
216 } else if (!matches(*argv
, "udp6zerocsumtx")) {
218 udp6zerocsumtx_set
= true;
219 } else if (!matches(*argv
, "noudp6zerocsumtx")) {
221 udp6zerocsumtx_set
= true;
222 } else if (!matches(*argv
, "udp6zerocsumrx")) {
224 udp6zerocsumrx_set
= true;
225 } else if (!matches(*argv
, "noudp6zerocsumrx")) {
227 udp6zerocsumrx_set
= true;
228 } else if (!matches(*argv
, "remcsumtx")) {
230 } else if (!matches(*argv
, "noremcsumtx")) {
232 } else if (!matches(*argv
, "remcsumrx")) {
234 } else if (!matches(*argv
, "noremcsumrx")) {
236 } else if (!matches(*argv
, "external")) {
239 } else if (!matches(*argv
, "noexternal")) {
241 } else if (!matches(*argv
, "gbp")) {
243 } else if (!matches(*argv
, "gpe")) {
245 } else if (matches(*argv
, "help") == 0) {
249 fprintf(stderr
, "vxlan: unknown command \"%s\"?\n", *argv
);
256 if (metadata
&& vni_set
) {
257 fprintf(stderr
, "vxlan: both 'external' and vni cannot be specified\n");
261 if (!metadata
&& !vni_set
) {
262 fprintf(stderr
, "vxlan: missing virtual network identifier\n");
266 if ((gaddr
&& daddr
) ||
267 (memcmp(&gaddr6
, &in6addr_any
, sizeof(gaddr6
)) &&
268 memcmp(&daddr6
, &in6addr_any
, sizeof(daddr6
)))) {
269 fprintf(stderr
, "vxlan: both group and remote cannot be specified\n");
273 if (!dst_port_set
&& gpe
) {
275 } else if (!dst_port_set
) {
276 fprintf(stderr
, "vxlan: destination port not specified\n"
277 "Will use Linux kernel default (non-standard value)\n");
279 "Use 'dstport 4789' to get the IANA assigned value\n"
280 "Use 'dstport 0' to get default and quiet this message\n");
283 addattr32(n
, 1024, IFLA_VXLAN_ID
, vni
);
285 addattr_l(n
, 1024, IFLA_VXLAN_GROUP
, &gaddr
, 4);
287 addattr_l(n
, 1024, IFLA_VXLAN_GROUP
, &daddr
, 4);
288 if (memcmp(&gaddr6
, &in6addr_any
, sizeof(gaddr6
)) != 0)
289 addattr_l(n
, 1024, IFLA_VXLAN_GROUP6
, &gaddr6
, sizeof(struct in6_addr
));
290 else if (memcmp(&daddr6
, &in6addr_any
, sizeof(daddr6
)) != 0)
291 addattr_l(n
, 1024, IFLA_VXLAN_GROUP6
, &daddr6
, sizeof(struct in6_addr
));
294 addattr_l(n
, 1024, IFLA_VXLAN_LOCAL
, &saddr
, 4);
295 else if (memcmp(&saddr6
, &in6addr_any
, sizeof(saddr6
)) != 0)
296 addattr_l(n
, 1024, IFLA_VXLAN_LOCAL6
, &saddr6
, sizeof(struct in6_addr
));
299 addattr32(n
, 1024, IFLA_VXLAN_LINK
, link
);
300 addattr32(n
, 1024, IFLA_VXLAN_LABEL
, label
);
301 addattr8(n
, 1024, IFLA_VXLAN_TTL
, ttl
);
302 addattr8(n
, 1024, IFLA_VXLAN_TOS
, tos
);
303 addattr8(n
, 1024, IFLA_VXLAN_LEARNING
, learning
);
304 addattr8(n
, 1024, IFLA_VXLAN_PROXY
, proxy
);
305 addattr8(n
, 1024, IFLA_VXLAN_RSC
, rsc
);
306 addattr8(n
, 1024, IFLA_VXLAN_L2MISS
, l2miss
);
307 addattr8(n
, 1024, IFLA_VXLAN_L3MISS
, l3miss
);
308 addattr8(n
, 1024, IFLA_VXLAN_REMCSUM_TX
, remcsumtx
);
309 addattr8(n
, 1024, IFLA_VXLAN_REMCSUM_RX
, remcsumrx
);
310 addattr8(n
, 1024, IFLA_VXLAN_COLLECT_METADATA
, metadata
);
313 addattr8(n
, 1024, IFLA_VXLAN_UDP_CSUM
, udpcsum
);
314 if (udp6zerocsumtx_set
)
315 addattr8(n
, 1024, IFLA_VXLAN_UDP_ZERO_CSUM6_TX
, udp6zerocsumtx
);
316 if (udp6zerocsumrx_set
)
317 addattr8(n
, 1024, IFLA_VXLAN_UDP_ZERO_CSUM6_RX
, udp6zerocsumrx
);
319 addattr32(n
, 1024, IFLA_VXLAN_AGEING
, 0);
321 addattr32(n
, 1024, IFLA_VXLAN_AGEING
, age
);
323 addattr32(n
, 1024, IFLA_VXLAN_LIMIT
, maxaddr
);
324 if (range
.low
|| range
.high
)
325 addattr_l(n
, 1024, IFLA_VXLAN_PORT_RANGE
,
326 &range
, sizeof(range
));
328 addattr16(n
, 1024, IFLA_VXLAN_PORT
, htons(dstport
));
331 addattr_l(n
, 1024, IFLA_VXLAN_GBP
, NULL
, 0);
333 addattr_l(n
, 1024, IFLA_VXLAN_GPE
, NULL
, 0);
339 static void vxlan_print_opt(struct link_util
*lu
, FILE *f
, struct rtattr
*tb
[])
350 if (!tb
[IFLA_VXLAN_ID
] ||
351 RTA_PAYLOAD(tb
[IFLA_VXLAN_ID
]) < sizeof(__u32
))
354 vni
= rta_getattr_u32(tb
[IFLA_VXLAN_ID
]);
355 fprintf(f
, "id %u ", vni
);
357 if (tb
[IFLA_VXLAN_GROUP
]) {
358 __be32 addr
= rta_getattr_u32(tb
[IFLA_VXLAN_GROUP
]);
361 if (IN_MULTICAST(ntohl(addr
)))
362 fprintf(f
, "group %s ",
363 format_host(AF_INET
, 4, &addr
));
365 fprintf(f
, "remote %s ",
366 format_host(AF_INET
, 4, &addr
));
368 } else if (tb
[IFLA_VXLAN_GROUP6
]) {
369 struct in6_addr addr
;
371 memcpy(&addr
, RTA_DATA(tb
[IFLA_VXLAN_GROUP6
]), sizeof(struct in6_addr
));
372 if (memcmp(&addr
, &in6addr_any
, sizeof(addr
)) != 0) {
373 if (IN6_IS_ADDR_MULTICAST(&addr
))
374 fprintf(f
, "group %s ",
375 format_host(AF_INET6
, sizeof(struct in6_addr
), &addr
));
377 fprintf(f
, "remote %s ",
378 format_host(AF_INET6
, sizeof(struct in6_addr
), &addr
));
382 if (tb
[IFLA_VXLAN_LOCAL
]) {
383 __be32 addr
= rta_getattr_u32(tb
[IFLA_VXLAN_LOCAL
]);
386 fprintf(f
, "local %s ",
387 format_host(AF_INET
, 4, &addr
));
388 } else if (tb
[IFLA_VXLAN_LOCAL6
]) {
389 struct in6_addr addr
;
391 memcpy(&addr
, RTA_DATA(tb
[IFLA_VXLAN_LOCAL6
]), sizeof(struct in6_addr
));
392 if (memcmp(&addr
, &in6addr_any
, sizeof(addr
)) != 0)
393 fprintf(f
, "local %s ",
394 format_host(AF_INET6
, sizeof(struct in6_addr
), &addr
));
397 if (tb
[IFLA_VXLAN_LINK
] &&
398 (link
= rta_getattr_u32(tb
[IFLA_VXLAN_LINK
]))) {
399 const char *n
= if_indextoname(link
, s2
);
402 fprintf(f
, "dev %s ", n
);
404 fprintf(f
, "dev %u ", link
);
407 if (tb
[IFLA_VXLAN_PORT_RANGE
]) {
408 const struct ifla_vxlan_port_range
*r
409 = RTA_DATA(tb
[IFLA_VXLAN_PORT_RANGE
]);
410 fprintf(f
, "srcport %u %u ", ntohs(r
->low
), ntohs(r
->high
));
413 if (tb
[IFLA_VXLAN_PORT
])
414 fprintf(f
, "dstport %u ",
415 ntohs(rta_getattr_u16(tb
[IFLA_VXLAN_PORT
])));
417 if (tb
[IFLA_VXLAN_LEARNING
] &&
418 !rta_getattr_u8(tb
[IFLA_VXLAN_LEARNING
]))
419 fputs("nolearning ", f
);
421 if (tb
[IFLA_VXLAN_PROXY
] && rta_getattr_u8(tb
[IFLA_VXLAN_PROXY
]))
424 if (tb
[IFLA_VXLAN_RSC
] && rta_getattr_u8(tb
[IFLA_VXLAN_RSC
]))
427 if (tb
[IFLA_VXLAN_L2MISS
] && rta_getattr_u8(tb
[IFLA_VXLAN_L2MISS
]))
430 if (tb
[IFLA_VXLAN_L3MISS
] && rta_getattr_u8(tb
[IFLA_VXLAN_L3MISS
]))
433 if (tb
[IFLA_VXLAN_TOS
] &&
434 (tos
= rta_getattr_u8(tb
[IFLA_VXLAN_TOS
]))) {
436 fprintf(f
, "tos inherit ");
438 fprintf(f
, "tos %#x ", tos
);
441 if (tb
[IFLA_VXLAN_TTL
]) {
442 __u8 ttl
= rta_getattr_u8(tb
[IFLA_VXLAN_TTL
]);
445 fprintf(f
, "ttl %d ", ttl
);
448 if (tb
[IFLA_VXLAN_LABEL
]) {
449 __u32 label
= rta_getattr_u32(tb
[IFLA_VXLAN_LABEL
]);
452 fprintf(f
, "flowlabel %#x ", ntohl(label
));
455 if (tb
[IFLA_VXLAN_AGEING
]) {
456 __u32 age
= rta_getattr_u32(tb
[IFLA_VXLAN_AGEING
]);
459 fprintf(f
, "ageing none ");
461 fprintf(f
, "ageing %u ", age
);
464 if (tb
[IFLA_VXLAN_LIMIT
] &&
465 ((maxaddr
= rta_getattr_u32(tb
[IFLA_VXLAN_LIMIT
])) != 0))
466 fprintf(f
, "maxaddr %u ", maxaddr
);
468 if (tb
[IFLA_VXLAN_UDP_CSUM
]) {
469 if (!rta_getattr_u8(tb
[IFLA_VXLAN_UDP_CSUM
]))
471 fputs("udpcsum ", f
);
474 if (tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_TX
]) {
475 if (!rta_getattr_u8(tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_TX
]))
477 fputs("udp6zerocsumtx ", f
);
480 if (tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_RX
]) {
481 if (!rta_getattr_u8(tb
[IFLA_VXLAN_UDP_ZERO_CSUM6_RX
]))
483 fputs("udp6zerocsumrx ", f
);
486 if (tb
[IFLA_VXLAN_REMCSUM_TX
] &&
487 rta_getattr_u8(tb
[IFLA_VXLAN_REMCSUM_TX
]))
488 fputs("remcsumtx ", f
);
490 if (tb
[IFLA_VXLAN_REMCSUM_RX
] &&
491 rta_getattr_u8(tb
[IFLA_VXLAN_REMCSUM_RX
]))
492 fputs("remcsumrx ", f
);
494 if (tb
[IFLA_VXLAN_COLLECT_METADATA
] &&
495 rta_getattr_u8(tb
[IFLA_VXLAN_COLLECT_METADATA
]))
496 fputs("external ", f
);
498 if (tb
[IFLA_VXLAN_GBP
])
500 if (tb
[IFLA_VXLAN_GPE
])
504 static void vxlan_print_help(struct link_util
*lu
, int argc
, char **argv
,
510 struct link_util vxlan_link_util
= {
512 .maxattr
= IFLA_VXLAN_MAX
,
513 .parse_opt
= vxlan_parse_opt
,
514 .print_opt
= vxlan_print_opt
,
515 .print_help
= vxlan_print_help
,