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
->vni
<= VXLAN_VID_MAX
) {
31 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_ID
, v
->vni
);
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
) == 0) {
37 if (v
->remote_family
== AF_INET
)
38 r
= sd_netlink_message_append_in_addr(m
, IFLA_VXLAN_GROUP
, &v
->remote
.in
);
40 r
= sd_netlink_message_append_in6_addr(m
, IFLA_VXLAN_GROUP6
, &v
->remote
.in6
);
42 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_GROUP attribute: %m");
45 if (in_addr_is_null(v
->local_family
, &v
->local
) == 0) {
46 if (v
->local_family
== AF_INET
)
47 r
= sd_netlink_message_append_in_addr(m
, IFLA_VXLAN_LOCAL
, &v
->local
.in
);
49 r
= sd_netlink_message_append_in6_addr(m
, IFLA_VXLAN_LOCAL6
, &v
->local
.in6
);
51 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LOCAL attribute: %m");
54 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_LINK
, link
->ifindex
);
56 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LINK attribute: %m");
59 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_TTL
, v
->ttl
);
61 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_TTL attribute: %m");
65 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_TOS
, v
->tos
);
67 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_TOS attribute: %m");
70 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_LEARNING
, v
->learning
);
72 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LEARNING attribute: %m");
74 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_RSC
, v
->route_short_circuit
);
76 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_RSC attribute: %m");
78 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_PROXY
, v
->arp_proxy
);
80 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_PROXY attribute: %m");
82 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_L2MISS
, v
->l2miss
);
84 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_L2MISS attribute: %m");
86 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_L3MISS
, v
->l3miss
);
88 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_L3MISS attribute: %m");
90 if (v
->fdb_ageing
!= 0) {
91 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_AGEING
, v
->fdb_ageing
/ USEC_PER_SEC
);
93 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_AGEING attribute: %m");
96 if (v
->max_fdb
!= 0) {
97 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_LIMIT
, v
->max_fdb
);
99 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LIMIT attribute: %m");
102 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_UDP_CSUM
, v
->udpcsum
);
104 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_UDP_CSUM attribute: %m");
106 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_UDP_ZERO_CSUM6_TX
, v
->udp6zerocsumtx
);
108 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_TX attribute: %m");
110 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_UDP_ZERO_CSUM6_RX
, v
->udp6zerocsumrx
);
112 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_UDP_ZERO_CSUM6_RX attribute: %m");
114 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_REMCSUM_TX
, v
->remote_csum_tx
);
116 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_REMCSUM_TX attribute: %m");
118 r
= sd_netlink_message_append_u8(m
, IFLA_VXLAN_REMCSUM_RX
, v
->remote_csum_rx
);
120 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_REMCSUM_RX attribute: %m");
122 r
= sd_netlink_message_append_u16(m
, IFLA_VXLAN_PORT
, htobe16(v
->dest_port
));
124 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_PORT attribute: %m");
126 if (v
->port_range
.low
!= 0 || v
->port_range
.high
!= 0) {
127 struct ifla_vxlan_port_range port_range
;
129 port_range
.low
= htobe16(v
->port_range
.low
);
130 port_range
.high
= htobe16(v
->port_range
.high
);
132 r
= sd_netlink_message_append_data(m
, IFLA_VXLAN_PORT_RANGE
, &port_range
, sizeof(port_range
));
134 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_PORT_RANGE attribute: %m");
137 r
= sd_netlink_message_append_u32(m
, IFLA_VXLAN_LABEL
, htobe32(v
->flow_label
));
139 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_LABEL attribute: %m");
141 if (v
->group_policy
) {
142 r
= sd_netlink_message_append_flag(m
, IFLA_VXLAN_GBP
);
144 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_GBP attribute: %m");
147 if (v
->generic_protocol_extension
) {
148 r
= sd_netlink_message_append_flag(m
, IFLA_VXLAN_GPE
);
150 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_VXLAN_GPE attribute: %m");
156 int config_parse_vxlan_address(const char *unit
,
157 const char *filename
,
160 unsigned section_line
,
167 union in_addr_union
*addr
= data
, buffer
;
175 r
= in_addr_from_string_auto(rvalue
, &f
, &buffer
);
177 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "vxlan '%s' address is invalid, ignoring assignment: %s", lvalue
, rvalue
);
181 r
= in_addr_is_multicast(f
, &buffer
);
183 if (streq(lvalue
, "Group")) {
185 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "vxlan %s invalid multicast address, ignoring assignment: %s", lvalue
, rvalue
);
192 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "vxlan %s cannot be a multicast address, ignoring assignment: %s", lvalue
, rvalue
);
196 if (streq(lvalue
, "Remote"))
197 v
->remote_family
= f
;
207 int config_parse_port_range(const char *unit
,
208 const char *filename
,
211 unsigned section_line
,
226 r
= parse_ip_port_range(rvalue
, &low
, &high
);
228 log_syntax(unit
, LOG_ERR
, filename
, line
, r
,
229 "Failed to parse VXLAN port range '%s'. Port should be greater than 0 and less than 65535.", rvalue
);
233 v
->port_range
.low
= low
;
234 v
->port_range
.high
= high
;
239 int config_parse_flow_label(const char *unit
,
240 const char *filename
,
243 unsigned section_line
,
258 r
= safe_atou(rvalue
, &f
);
260 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Failed to parse VXLAN flow label '%s'.", rvalue
);
264 if (f
& ~VXLAN_FLOW_LABEL_MAX_MASK
) {
265 log_syntax(unit
, LOG_ERR
, filename
, line
, r
,
266 "VXLAN flow label '%s' not valid. Flow label range should be [0-1048575].", rvalue
);
275 static int netdev_vxlan_verify(NetDev
*netdev
, const char *filename
) {
276 VxLan
*v
= VXLAN(netdev
);
282 if (v
->vni
> VXLAN_VID_MAX
)
283 return log_netdev_warning_errno(netdev
, SYNTHETIC_ERRNO(EINVAL
),
284 "%s: VXLAN without valid VNI (or VXLAN Segment ID) configured. Ignoring.",
288 return log_netdev_warning_errno(netdev
, SYNTHETIC_ERRNO(EINVAL
),
289 "%s: VXLAN TTL must be <= 255. Ignoring.",
292 if (!v
->dest_port
&& v
->generic_protocol_extension
)
298 static void vxlan_init(NetDev
*netdev
) {
307 v
->vni
= VXLAN_VID_MAX
+ 1;
310 v
->udp6zerocsumtx
= false;
311 v
->udp6zerocsumrx
= false;
314 const NetDevVTable vxlan_vtable
= {
315 .object_size
= sizeof(VxLan
),
317 .sections
= "Match\0NetDev\0VXLAN\0",
318 .fill_message_create
= netdev_vxlan_fill_message_create
,
319 .create_type
= NETDEV_CREATE_STACKED
,
320 .config_verify
= netdev_vxlan_verify
,