1 /* SPDX-License-Identifier: LGPL-2.1+ */
5 #include "sd-netlink.h"
7 #include "alloc-util.h"
8 #include "conf-parser.h"
9 #include "extract-word.h"
11 #include "netlink-util.h"
12 #include "parse-util.h"
13 #include "string-util.h"
16 #include "networkd-manager.h"
18 #define GENEVE_FLOW_LABEL_MAX_MASK 0xFFFFFU
19 #define DEFAULT_GENEVE_DESTINATION_PORT 6081
21 /* callback for geneve netdev's created without a backing Link */
22 static int geneve_netdev_create_handler(sd_netlink
*rtnl
, sd_netlink_message
*m
, NetDev
*netdev
) {
26 assert(netdev
->state
!= _NETDEV_STATE_INVALID
);
28 r
= sd_netlink_message_get_errno(m
);
30 log_netdev_info(netdev
, "Geneve netdev exists, using existing without changing its parameters");
32 log_netdev_warning_errno(netdev
, r
, "Geneve netdev could not be created: %m");
38 log_netdev_debug(netdev
, "Geneve created");
43 static int netdev_geneve_create(NetDev
*netdev
) {
44 _cleanup_(sd_netlink_message_unrefp
) sd_netlink_message
*m
= NULL
;
52 r
= sd_rtnl_message_new_link(netdev
->manager
->rtnl
, &m
, RTM_NEWLINK
, 0);
54 return log_netdev_error_errno(netdev
, r
, "Could not allocate RTM_NEWLINK message: %m");
56 r
= sd_netlink_message_append_string(m
, IFLA_IFNAME
, netdev
->ifname
);
58 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_IFNAME, attribute: %m");
61 r
= sd_netlink_message_append_ether_addr(m
, IFLA_ADDRESS
, netdev
->mac
);
63 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_ADDRESS attribute: %m");
66 if (netdev
->mtu
!= 0) {
67 r
= sd_netlink_message_append_u32(m
, IFLA_MTU
, netdev
->mtu
);
69 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_MTU attribute: %m");
72 r
= sd_netlink_message_open_container(m
, IFLA_LINKINFO
);
74 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
76 r
= sd_netlink_message_open_container_union(m
, IFLA_INFO_DATA
, netdev_kind_to_string(netdev
->kind
));
78 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
80 if (v
->id
<= GENEVE_VID_MAX
) {
81 r
= sd_netlink_message_append_u32(m
, IFLA_GENEVE_ID
, v
->id
);
83 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_ID attribute: %m");
86 if (in_addr_is_null(v
->remote_family
, &v
->remote
) == 0) {
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
);
92 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_REMOTE/IFLA_GENEVE_REMOTE6 attribute: %m");
96 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_TTL
, v
->ttl
);
98 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_TTL attribute: %m");
101 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_TOS
, v
->tos
);
103 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_TOS attribute: %m");
105 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_UDP_CSUM
, v
->udpcsum
);
107 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_UDP_CSUM attribute: %m");
109 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_UDP_ZERO_CSUM6_TX
, v
->udp6zerocsumtx
);
111 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_UDP_ZERO_CSUM6_TX attribute: %m");
113 r
= sd_netlink_message_append_u8(m
, IFLA_GENEVE_UDP_ZERO_CSUM6_RX
, v
->udp6zerocsumrx
);
115 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_UDP_ZERO_CSUM6_RX attribute: %m");
117 if (v
->dest_port
!= DEFAULT_GENEVE_DESTINATION_PORT
) {
118 r
= sd_netlink_message_append_u16(m
, IFLA_GENEVE_PORT
, htobe16(v
->dest_port
));
120 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_PORT attribute: %m");
123 if (v
->flow_label
> 0) {
124 r
= sd_netlink_message_append_u32(m
, IFLA_GENEVE_LABEL
, htobe32(v
->flow_label
));
126 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_GENEVE_LABEL attribute: %m");
129 r
= sd_netlink_message_close_container(m
);
131 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
133 r
= sd_netlink_message_close_container(m
);
135 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
137 r
= netlink_call_async(netdev
->manager
->rtnl
, NULL
, m
, geneve_netdev_create_handler
,
138 netdev_destroy_callback
, netdev
);
140 return log_netdev_error_errno(netdev
, r
, "Could not send rtnetlink message: %m");
143 netdev
->state
= NETDEV_STATE_CREATING
;
145 log_netdev_debug(netdev
, "Creating");
150 int config_parse_geneve_vni(const char *unit
,
151 const char *filename
,
154 unsigned section_line
,
160 Geneve
*v
= userdata
;
169 r
= safe_atou32(rvalue
, &f
);
171 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Failed to parse Geneve VNI '%s'.", rvalue
);
175 if (f
> GENEVE_VID_MAX
){
176 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Geneve VNI out is of range '%s'.", rvalue
);
185 int config_parse_geneve_address(const char *unit
,
186 const char *filename
,
189 unsigned section_line
,
195 Geneve
*v
= userdata
;
196 union in_addr_union
*addr
= data
, buffer
;
204 r
= in_addr_from_string_auto(rvalue
, &f
, &buffer
);
206 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "geneve '%s' address is invalid, ignoring assignment: %s", lvalue
, rvalue
);
210 r
= in_addr_is_multicast(f
, &buffer
);
212 log_syntax(unit
, LOG_ERR
, filename
, line
, 0, "geneve invalid multicast '%s' address, ignoring assignment: %s", lvalue
, rvalue
);
216 v
->remote_family
= f
;
222 int config_parse_geneve_flow_label(const char *unit
,
223 const char *filename
,
226 unsigned section_line
,
232 Geneve
*v
= userdata
;
241 r
= safe_atou32(rvalue
, &f
);
243 log_syntax(unit
, LOG_ERR
, filename
, line
, r
, "Failed to parse Geneve flow label '%s'.", rvalue
);
247 if (f
& ~GENEVE_FLOW_LABEL_MAX_MASK
) {
248 log_syntax(unit
, LOG_ERR
, filename
, line
, r
,
249 "Geneve flow label '%s' not valid. Flow label range should be [0-1048575].", rvalue
);
258 static int netdev_geneve_verify(NetDev
*netdev
, const char *filename
) {
259 Geneve
*v
= GENEVE(netdev
);
266 log_warning("Invalid Geneve TTL value '0' configured in '%s'. Ignoring", filename
);
273 static void geneve_init(NetDev
*netdev
) {
282 v
->id
= GENEVE_VID_MAX
+ 1;
283 v
->dest_port
= DEFAULT_GENEVE_DESTINATION_PORT
;
285 v
->udp6zerocsumtx
= false;
286 v
->udp6zerocsumrx
= false;
289 const NetDevVTable geneve_vtable
= {
290 .object_size
= sizeof(Geneve
),
292 .sections
= "Match\0NetDev\0GENEVE\0",
293 .create
= netdev_geneve_create
,
294 .create_type
= NETDEV_CREATE_INDEPENDENT
,
295 .config_verify
= netdev_geneve_verify
,