1 /* SPDX-License-Identifier: LGPL-2.1+ */
5 #include "sd-netlink.h"
7 #include "conf-parser.h"
8 #include "alloc-util.h"
9 #include "extract-word.h"
10 #include "string-util.h"
12 #include "parse-util.h"
15 #include "networkd-link.h"
16 #include "netdev/vxlan.h"
18 static int netdev_vxlan_fill_message_create(NetDev
*netdev
, Link
*link
, sd_netlink_message
*m
) {
30 if (v
->id
<= VXLAN_VID_MAX
) {
31 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_ID
, v
->id
);
33 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_ID attribute: %m");
36 if (!in_addr_is_null(v
->remote_family
, &v
->remote
)) {
38 if (v
->remote_family
== AF_INET
)
39 r
= sd_netlink_message_append_in_addr(m
, IFLA_VXLAN_GROUP
, &v
->remote
.in
);
41 r
= sd_netlink_message_append_in6_addr(m
, IFLA_VXLAN_GROUP6
, &v
->remote
.in6
);
44 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_GROUP attribute: %m");
47 if (!in_addr_is_null(v
->local_family
, &v
->local
)) {
49 if (v
->local_family
== AF_INET
)
50 r
= sd_netlink_message_append_in_addr(m
, IFLA_VXLAN_LOCAL
, &v
->local
.in
);
52 r
= sd_netlink_message_append_in6_addr(m
, IFLA_VXLAN_LOCAL6
, &v
->local
.in6
);
55 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LOCAL attribute: %m");
58 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_LINK
, link
->ifindex
);
60 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LINK attribute: %m");
63 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_TTL
, v
->ttl
);
65 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_TTL attribute: %m");
69 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_TOS
, v
->tos
);
71 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_TOS attribute: %m");
74 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_LEARNING
, v
->learning
);
76 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LEARNING attribute: %m");
78 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_RSC
, v
->route_short_circuit
);
80 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_RSC attribute: %m");
82 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_PROXY
, v
->arp_proxy
);
84 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_PROXY attribute: %m");
86 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_L2MISS
, v
->l2miss
);
88 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_L2MISS attribute: %m");
90 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_L3MISS
, v
->l3miss
);
92 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_L3MISS attribute: %m");
94 if (v
->fdb_ageing
!= 0) {
95 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_AGEING
, v
->fdb_ageing
/ USEC_PER_SEC
);
97 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_AGEING attribute: %m");
100 if (v
->max_fdb
!= 0) {
101 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_LIMIT
, v
->max_fdb
);
103 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LIMIT attribute: %m");
106 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_UDP_CSUM
, v
->udpcsum
);
108 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_UDP_CSUM attribute: %m");
110 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_UDP_ZERO_CSUM6_TX
, v
->udp6zerocsumtx
);
112 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_TX attribute: %m");
114 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_UDP_ZERO_CSUM6_RX
, v
->udp6zerocsumrx
);
116 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_RX attribute: %m");
118 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_REMCSUM_TX
, v
->remote_csum_tx
);
120 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_REMCSUM_TX attribute: %m");
122 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_REMCSUM_RX
, v
->remote_csum_rx
);
124 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_REMCSUM_RX attribute: %m");
126 r
= sd_netlink_message_append_u16(m
, IFLA_VXLAN_PORT
, htobe16(v
->dest_port
));
128 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_PORT attribute: %m");
130 if (v
->port_range
.low
!= 0 || v
->port_range
.high
!= 0) {
131 struct ifla_vxlan_port_range port_range
;
133 port_range
.low
= htobe16(v
->port_range
.low
);
134 port_range
.high
= htobe16(v
->port_range
.high
);
136 r
= sd_netlink_message_append_data(m
, IFLA_VXLAN_PORT_RANGE
, &port_range
, sizeof(port_range
));
138 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_PORT_RANGE attribute: %m");
141 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_LABEL
, htobe32(v
->flow_label
));
143 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LABEL attribute: %m");
145 if (v
->group_policy
) {
146 r
= sd_netlink_message_append_flag(m
, IFLA_VXLAN_GBP
);
148 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_GBP attribute: %m");
154 int config_parse_vxlan_address(const char *unit
,
155 const char *filename
,
158 unsigned section_line
,
165 union in_addr_union
*addr
= data
, buffer
;
173 r
= in_addr_from_string_auto(rvalue
, &f
, &buffer
);
175 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "vxlan '%s' address is invalid, ignoring assignment: %s", lvalue
, rvalue
);
179 r
= in_addr_is_multicast(f
, &buffer
);
181 if (streq(lvalue
, "Group")) {
183 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "vxlan %s invalid multicast address, ignoring assignment: %s", lvalue
, rvalue
);
187 v
->remote_family
= f
;
190 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "vxlan %s cannot be a multicast address, ignoring assignment: %s", lvalue
, rvalue
);
194 if (streq(lvalue
, "Remote"))
195 v
->remote_family
= f
;
205 int config_parse_port_range(const char *unit
,
206 const char *filename
,
209 unsigned section_line
,
224 r
= parse_ip_port_range(rvalue
, &low
, &high
);
226 log_syntax(unit
, LOG_ERR
, filename
, line
, r
,
227 "Failed to parse VXLAN port range '%s'. Port should be greater than 0 and less than 65535.", rvalue
);
231 v
->port_range
.low
= low
;
232 v
->port_range
.high
= high
;
237 int config_parse_flow_label(const char *unit
,
238 const char *filename
,
241 unsigned section_line
,
256 r
= safe_atou(rvalue
, &f
);
258 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Failed to parse VXLAN flow label '%s'.", rvalue
);
262 if (f
& ~VXLAN_FLOW_LABEL_MAX_MASK
) {
263 log_syntax(unit
, LOG_ERR
, filename
, line
, r
,
264 "VXLAN flow label '%s' not valid. Flow label range should be [0-1048575].", rvalue
);
273 static int netdev_vxlan_verify(NetDev
*netdev
, const char *filename
) {
274 VxLan
*v
= VXLAN(netdev
);
280 if (v
->id
> VXLAN_VID_MAX
) {
281 log_warning("VXLAN without valid Id configured in %s. Ignoring", filename
);
288 static void vxlan_init(NetDev
*netdev
) {
297 v
->id
= VXLAN_VID_MAX
+ 1;
300 v
->udp6zerocsumtx
= false;
301 v
->udp6zerocsumrx
= false;
304 const NetDevVTable vxlan_vtable
= {
305 .object_size
= sizeof(VxLan
),
307 .sections
= "Match\0NetDev\0VXLAN\0",
308 .fill_message_create
= netdev_vxlan_fill_message_create
,
309 .create_type
= NETDEV_CREATE_STACKED
,
310 .config_verify
= netdev_vxlan_verify
,