1 /* SPDX-License-Identifier: LGPL-2.1+ */
5 #include "alloc-util.h"
6 #include "conf-parser.h"
7 #include "extract-word.h"
9 #include "parse-util.h"
10 #include "sd-netlink.h"
11 #include "string-util.h"
14 #include "networkd-manager.h"
16 #define GENEVE_FLOW_LABEL_MAX_MASK 0xFFFFFU
17 #define DEFAULT_GENEVE_DESTINATION_PORT 6081
19 /* callback for geneve netdev's created without a backing Link */
20 static int geneve_netdev_create_handler(sd_netlink
*rtnl
, sd_netlink_message
*m
, void *userdata
) {
21 NetDev
*netdev
= userdata
;
25 assert(netdev
->state
!= _NETDEV_STATE_INVALID
);
27 r
= sd_netlink_message_get_errno(m
);
29 log_netdev_info(netdev
, "Geneve netdev exists, using existing without changing its parameters");
31 log_netdev_warning_errno(netdev
, r
, "Geneve netdev could not be created: %m");
37 log_netdev_debug(netdev
, "Geneve created");
42 static int netdev_geneve_create(NetDev
*netdev
) {
43 _cleanup_(sd_netlink_message_unrefp
) sd_netlink_message
*m
= NULL
;
51 r
= sd_rtnl_message_new_link(netdev
->manager
->rtnl
, &m
, RTM_NEWLINK
, 0);
53 return log_netdev_error_errno(netdev
, r
, "Could not allocate RTM_NEWLINK message: %m");
55 r
= sd_netlink_message_append_string(m
, IFLA_IFNAME
, netdev
->ifname
);
57 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_IFNAME, attribute: %m");
60 r
= sd_netlink_message_append_ether_addr(m
, IFLA_ADDRESS
, netdev
->mac
);
62 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_ADDRESS attribute: %m");
65 if (netdev
->mtu
!= 0) {
66 r
= sd_netlink_message_append_u32(m
, IFLA_MTU
, netdev
->mtu
);
68 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_MTU attribute: %m");
71 r
= sd_netlink_message_open_container(m
, IFLA_LINKINFO
);
73 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
75 r
= sd_netlink_message_open_container_union(m
, IFLA_INFO_DATA
, netdev_kind_to_string(netdev
->kind
));
77 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
79 if (v
->id
<= GENEVE_VID_MAX
) {
80 r
= sd_netlink_message_append_u32(m
, IFLA_GENEVE_ID
, v
->id
);
82 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_ID attribute: %m");
85 if (!in_addr_is_null(v
->remote_family
, &v
->remote
)) {
87 if (v
->remote_family
== AF_INET
)
88 r
= sd_netlink_message_append_in_addr(m
, IFLA_GENEVE_REMOTE
, &v
->remote
.in
);
90 r
= sd_netlink_message_append_in6_addr(m
, IFLA_GENEVE_REMOTE6
, &v
->remote
.in6
);
93 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_GROUP attribute: %m");
98 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_TTL
, v
->ttl
);
100 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_TTL attribute: %m");
103 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_TOS
, v
->tos
);
105 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_TOS attribute: %m");
107 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_UDP_CSUM
, v
->udpcsum
);
109 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_UDP_CSUM attribute: %m");
111 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_UDP_ZERO_CSUM6_TX
, v
->udp6zerocsumtx
);
113 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_UDP_ZERO_CSUM6_TX attribute: %m");
115 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_UDP_ZERO_CSUM6_RX
, v
->udp6zerocsumrx
);
117 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_UDP_ZERO_CSUM6_RX attribute: %m");
119 if (v
->dest_port
!= DEFAULT_GENEVE_DESTINATION_PORT
) {
120 r
= sd_netlink_message_append_u16(m
, IFLA_GENEVE_PORT
, htobe16(v
->dest_port
));
122 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_PORT attribute: %m");
125 if (v
->flow_label
> 0) {
126 r
= sd_netlink_message_append_u32(m
, IFLA_GENEVE_LABEL
, htobe32(v
->flow_label
));
128 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_LABEL attribute: %m");
131 r
= sd_netlink_message_close_container(m
);
133 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
135 r
= sd_netlink_message_close_container(m
);
137 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
139 r
= sd_netlink_call_async(netdev
->manager
->rtnl
, m
, geneve_netdev_create_handler
,
140 netdev_netlink_destroy_callback
, netdev
, 0, NULL
);
142 return log_netdev_error_errno(netdev
, r
, "Could not send rtnetlink message: %m");
145 netdev
->state
= NETDEV_STATE_CREATING
;
147 log_netdev_debug(netdev
, "Creating");
152 int config_parse_geneve_vni(const char *unit
,
153 const char *filename
,
156 unsigned section_line
,
162 Geneve
*v
= userdata
;
171 r
= safe_atou32(rvalue
, &f
);
173 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Failed to parse Geneve VNI '%s'.", rvalue
);
177 if (f
> GENEVE_VID_MAX
){
178 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Geneve VNI out is of range '%s'.", rvalue
);
187 int config_parse_geneve_address(const char *unit
,
188 const char *filename
,
191 unsigned section_line
,
197 Geneve
*v
= userdata
;
198 union in_addr_union
*addr
= data
, buffer
;
206 r
= in_addr_from_string_auto(rvalue
, &f
, &buffer
);
208 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "geneve '%s' address is invalid, ignoring assignment: %s", lvalue
, rvalue
);
212 r
= in_addr_is_multicast(f
, &buffer
);
214 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "geneve invalid multicast '%s' address, ignoring assignment: %s", lvalue
, rvalue
);
218 v
->remote_family
= f
;
224 int config_parse_geneve_flow_label(const char *unit
,
225 const char *filename
,
228 unsigned section_line
,
234 Geneve
*v
= userdata
;
243 r
= safe_atou32(rvalue
, &f
);
245 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Failed to parse Geneve flow label '%s'.", rvalue
);
249 if (f
& ~GENEVE_FLOW_LABEL_MAX_MASK
) {
250 log_syntax(unit
, LOG_ERR
, filename
, line
, r
,
251 "Geneve flow label '%s' not valid. Flow label range should be [0-1048575].", rvalue
);
260 static int netdev_geneve_verify(NetDev
*netdev
, const char *filename
) {
261 Geneve
*v
= GENEVE(netdev
);
268 log_warning("Invalid Geneve TTL value '0' configured in '%s'. Ignoring", filename
);
275 static void geneve_init(NetDev
*netdev
) {
284 v
->id
= GENEVE_VID_MAX
+ 1;
285 v
->dest_port
= DEFAULT_GENEVE_DESTINATION_PORT
;
287 v
->udp6zerocsumtx
= false;
288 v
->udp6zerocsumrx
= false;
291 const NetDevVTable geneve_vtable
= {
292 .object_size
= sizeof(Geneve
),
294 .sections
= "Match\0NetDev\0GENEVE\0",
295 .create
= netdev_geneve_create
,
296 .create_type
= NETDEV_CREATE_INDEPENDENT
,
297 .config_verify
= netdev_geneve_verify
,