1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2013 Tom Gundersen <teg@jklm.no>
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
24 #include "conf-files.h"
25 #include "conf-parser.h"
27 #include "netlink-util.h"
28 #include "network-internal.h"
30 #include "siphash24.h"
31 #include "string-util.h"
32 #include "networkd-netdev.h"
34 const NetDevVTable
* const netdev_vtable
[_NETDEV_KIND_MAX
] = {
36 [NETDEV_KIND_BRIDGE
] = &bridge_vtable
,
37 [NETDEV_KIND_BOND
] = &bond_vtable
,
38 [NETDEV_KIND_VLAN
] = &vlan_vtable
,
39 [NETDEV_KIND_MACVLAN
] = &macvlan_vtable
,
40 [NETDEV_KIND_MACVTAP
] = &macvtap_vtable
,
41 [NETDEV_KIND_IPVLAN
] = &ipvlan_vtable
,
42 [NETDEV_KIND_VXLAN
] = &vxlan_vtable
,
43 [NETDEV_KIND_IPIP
] = &ipip_vtable
,
44 [NETDEV_KIND_GRE
] = &gre_vtable
,
45 [NETDEV_KIND_GRETAP
] = &gretap_vtable
,
46 [NETDEV_KIND_IP6GRE
] = &ip6gre_vtable
,
47 [NETDEV_KIND_IP6GRETAP
] = &ip6gretap_vtable
,
48 [NETDEV_KIND_SIT
] = &sit_vtable
,
49 [NETDEV_KIND_VTI
] = &vti_vtable
,
50 [NETDEV_KIND_VTI6
] = &vti6_vtable
,
51 [NETDEV_KIND_VETH
] = &veth_vtable
,
52 [NETDEV_KIND_DUMMY
] = &dummy_vtable
,
53 [NETDEV_KIND_TUN
] = &tun_vtable
,
54 [NETDEV_KIND_TAP
] = &tap_vtable
,
55 [NETDEV_KIND_IP6TNL
] = &ip6tnl_vtable
,
58 static const char* const netdev_kind_table
[_NETDEV_KIND_MAX
] = {
59 [NETDEV_KIND_BRIDGE
] = "bridge",
60 [NETDEV_KIND_BOND
] = "bond",
61 [NETDEV_KIND_VLAN
] = "vlan",
62 [NETDEV_KIND_MACVLAN
] = "macvlan",
63 [NETDEV_KIND_MACVTAP
] = "macvtap",
64 [NETDEV_KIND_IPVLAN
] = "ipvlan",
65 [NETDEV_KIND_VXLAN
] = "vxlan",
66 [NETDEV_KIND_IPIP
] = "ipip",
67 [NETDEV_KIND_GRE
] = "gre",
68 [NETDEV_KIND_GRETAP
] = "gretap",
69 [NETDEV_KIND_IP6GRE
] = "ip6gre",
70 [NETDEV_KIND_IP6GRETAP
] = "ip6gretap",
71 [NETDEV_KIND_SIT
] = "sit",
72 [NETDEV_KIND_VETH
] = "veth",
73 [NETDEV_KIND_VTI
] = "vti",
74 [NETDEV_KIND_VTI6
] = "vti6",
75 [NETDEV_KIND_DUMMY
] = "dummy",
76 [NETDEV_KIND_TUN
] = "tun",
77 [NETDEV_KIND_TAP
] = "tap",
78 [NETDEV_KIND_IP6TNL
] = "ip6tnl",
81 DEFINE_STRING_TABLE_LOOKUP(netdev_kind
, NetDevKind
);
82 DEFINE_CONFIG_PARSE_ENUM(config_parse_netdev_kind
, netdev_kind
, NetDevKind
, "Failed to parse netdev kind");
84 static void netdev_cancel_callbacks(NetDev
*netdev
) {
85 _cleanup_netlink_message_unref_ sd_netlink_message
*m
= NULL
;
86 netdev_join_callback
*callback
;
91 rtnl_message_new_synthetic_error(-ENODEV
, 0, &m
);
93 while ((callback
= netdev
->callbacks
)) {
95 assert(callback
->link
);
96 assert(callback
->callback
);
97 assert(netdev
->manager
);
98 assert(netdev
->manager
->rtnl
);
100 callback
->callback(netdev
->manager
->rtnl
, m
, callback
->link
);
103 LIST_REMOVE(callbacks
, netdev
->callbacks
, callback
);
104 link_unref(callback
->link
);
109 static void netdev_free(NetDev
*netdev
) {
113 netdev_cancel_callbacks(netdev
);
116 hashmap_remove(netdev
->manager
->netdevs
, netdev
->ifname
);
118 free(netdev
->filename
);
120 free(netdev
->description
);
121 free(netdev
->ifname
);
124 condition_free_list(netdev
->match_host
);
125 condition_free_list(netdev
->match_virt
);
126 condition_free_list(netdev
->match_kernel
);
127 condition_free_list(netdev
->match_arch
);
129 if (NETDEV_VTABLE(netdev
) &&
130 NETDEV_VTABLE(netdev
)->done
)
131 NETDEV_VTABLE(netdev
)->done(netdev
);
136 NetDev
*netdev_unref(NetDev
*netdev
) {
137 if (netdev
&& (-- netdev
->n_ref
<= 0))
143 NetDev
*netdev_ref(NetDev
*netdev
) {
145 assert_se(++ netdev
->n_ref
>= 2);
150 void netdev_drop(NetDev
*netdev
) {
151 if (!netdev
|| netdev
->state
== NETDEV_STATE_LINGER
)
154 netdev
->state
= NETDEV_STATE_LINGER
;
156 log_netdev_debug(netdev
, "netdev removed");
158 netdev_cancel_callbacks(netdev
);
160 netdev_unref(netdev
);
165 int netdev_get(Manager
*manager
, const char *name
, NetDev
**ret
) {
172 netdev
= hashmap_get(manager
->netdevs
, name
);
183 static int netdev_enter_failed(NetDev
*netdev
) {
184 netdev
->state
= NETDEV_STATE_FAILED
;
186 netdev_cancel_callbacks(netdev
);
191 static int netdev_enslave_ready(NetDev
*netdev
, Link
* link
, sd_netlink_message_handler_t callback
) {
192 _cleanup_netlink_message_unref_ sd_netlink_message
*req
= NULL
;
196 assert(netdev
->state
== NETDEV_STATE_READY
);
197 assert(netdev
->manager
);
198 assert(netdev
->manager
->rtnl
);
199 assert(IN_SET(netdev
->kind
, NETDEV_KIND_BRIDGE
, NETDEV_KIND_BOND
));
203 r
= sd_rtnl_message_new_link(netdev
->manager
->rtnl
, &req
, RTM_SETLINK
, link
->ifindex
);
205 return log_netdev_error_errno(netdev
, r
, "Could not allocate RTM_SETLINK message: %m");
207 r
= sd_netlink_message_append_u32(req
, IFLA_MASTER
, netdev
->ifindex
);
209 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_MASTER attribute: %m");
211 r
= sd_netlink_call_async(netdev
->manager
->rtnl
, req
, callback
, link
, 0, NULL
);
213 return log_netdev_error(netdev
, "Could not send rtnetlink message: %m");
217 log_netdev_debug(netdev
, "Enslaving link '%s'", link
->ifname
);
222 static int netdev_enter_ready(NetDev
*netdev
) {
223 netdev_join_callback
*callback
, *callback_next
;
227 assert(netdev
->ifname
);
229 if (netdev
->state
!= NETDEV_STATE_CREATING
)
232 netdev
->state
= NETDEV_STATE_READY
;
234 log_netdev_info(netdev
, "netdev ready");
236 LIST_FOREACH_SAFE(callbacks
, callback
, callback_next
, netdev
->callbacks
) {
237 /* enslave the links that were attempted to be enslaved before the
239 r
= netdev_enslave_ready(netdev
, callback
->link
, callback
->callback
);
243 LIST_REMOVE(callbacks
, netdev
->callbacks
, callback
);
244 link_unref(callback
->link
);
248 if (NETDEV_VTABLE(netdev
)->post_create
)
249 NETDEV_VTABLE(netdev
)->post_create(netdev
, NULL
, NULL
);
254 /* callback for netdev's created without a backing Link */
255 static int netdev_create_handler(sd_netlink
*rtnl
, sd_netlink_message
*m
, void *userdata
) {
256 _cleanup_netdev_unref_ NetDev
*netdev
= userdata
;
259 assert(netdev
->state
!= _NETDEV_STATE_INVALID
);
261 r
= sd_netlink_message_get_errno(m
);
263 log_netdev_info(netdev
, "netdev exists, using existing without changing its parameters");
265 log_netdev_warning_errno(netdev
, r
, "netdev could not be created: %m");
271 log_netdev_debug(netdev
, "Created");
276 int netdev_enslave(NetDev
*netdev
, Link
*link
, sd_netlink_message_handler_t callback
) {
280 assert(netdev
->manager
);
281 assert(netdev
->manager
->rtnl
);
282 assert(IN_SET(netdev
->kind
, NETDEV_KIND_BRIDGE
, NETDEV_KIND_BOND
));
284 if (netdev
->state
== NETDEV_STATE_READY
) {
285 r
= netdev_enslave_ready(netdev
, link
, callback
);
288 } else if (IN_SET(netdev
->state
, NETDEV_STATE_LINGER
, NETDEV_STATE_FAILED
)) {
289 _cleanup_netlink_message_unref_ sd_netlink_message
*m
= NULL
;
291 r
= rtnl_message_new_synthetic_error(-ENODEV
, 0, &m
);
293 callback(netdev
->manager
->rtnl
, m
, link
);
295 /* the netdev is not yet read, save this request for when it is */
296 netdev_join_callback
*cb
;
298 cb
= new0(netdev_join_callback
, 1);
302 cb
->callback
= callback
;
306 LIST_PREPEND(callbacks
, netdev
->callbacks
, cb
);
308 log_netdev_debug(netdev
, "Will enslave '%s', when ready", link
->ifname
);
314 int netdev_set_ifindex(NetDev
*netdev
, sd_netlink_message
*message
) {
317 const char *received_kind
;
318 const char *received_name
;
324 r
= sd_netlink_message_get_type(message
, &type
);
326 return log_netdev_error_errno(netdev
, r
, "Could not get rtnl message type: %m");
328 if (type
!= RTM_NEWLINK
) {
329 log_netdev_error(netdev
, "Cannot set ifindex from unexpected rtnl message type.");
333 r
= sd_rtnl_message_link_get_ifindex(message
, &ifindex
);
335 log_netdev_error_errno(netdev
, r
, "Could not get ifindex: %m");
336 netdev_enter_failed(netdev
);
338 } else if (ifindex
<= 0) {
339 log_netdev_error(netdev
, "Got invalid ifindex: %d", ifindex
);
340 netdev_enter_failed(netdev
);
344 if (netdev
->ifindex
> 0) {
345 if (netdev
->ifindex
!= ifindex
) {
346 log_netdev_error(netdev
, "Could not set ifindex to %d, already set to %d",
347 ifindex
, netdev
->ifindex
);
348 netdev_enter_failed(netdev
);
351 /* ifindex already set to the same for this netdev */
355 r
= sd_netlink_message_read_string(message
, IFLA_IFNAME
, &received_name
);
357 return log_netdev_error_errno(netdev
, r
, "Could not get IFNAME: %m");
359 if (!streq(netdev
->ifname
, received_name
)) {
360 log_netdev_error(netdev
, "Received newlink with wrong IFNAME %s", received_name
);
361 netdev_enter_failed(netdev
);
365 r
= sd_netlink_message_enter_container(message
, IFLA_LINKINFO
);
367 return log_netdev_error_errno(netdev
, r
, "Could not get LINKINFO: %m");
369 r
= sd_netlink_message_read_string(message
, IFLA_INFO_KIND
, &received_kind
);
371 return log_netdev_error_errno(netdev
, r
, "Could not get KIND: %m");
373 r
= sd_netlink_message_exit_container(message
);
375 return log_netdev_error_errno(netdev
, r
, "Could not exit container: %m");
377 if (netdev
->kind
== NETDEV_KIND_TAP
)
378 /* the kernel does not distinguish between tun and tap */
381 kind
= netdev_kind_to_string(netdev
->kind
);
383 log_netdev_error(netdev
, "Could not get kind");
384 netdev_enter_failed(netdev
);
389 if (!streq(kind
, received_kind
)) {
390 log_netdev_error(netdev
,
391 "Received newlink with wrong KIND %s, "
392 "expected %s", received_kind
, kind
);
393 netdev_enter_failed(netdev
);
397 netdev
->ifindex
= ifindex
;
399 log_netdev_debug(netdev
, "netdev has index %d", netdev
->ifindex
);
401 netdev_enter_ready(netdev
);
406 #define HASH_KEY SD_ID128_MAKE(52,e1,45,bd,00,6f,29,96,21,c6,30,6d,83,71,04,48)
408 int netdev_get_mac(const char *ifname
, struct ether_addr
**ret
) {
409 _cleanup_free_
struct ether_addr
*mac
= NULL
;
418 mac
= new0(struct ether_addr
, 1);
423 sz
= sizeof(sd_id128_t
) + l
;
426 /* fetch some persistent data unique to the machine */
427 r
= sd_id128_get_machine((sd_id128_t
*) v
);
431 /* combine with some data unique (on this machine) to this
433 memcpy(v
+ sizeof(sd_id128_t
), ifname
, l
);
435 /* Let's hash the host machine ID plus the container name. We
436 * use a fixed, but originally randomly created hash key here. */
437 siphash24(result
, v
, sz
, HASH_KEY
.bytes
);
439 assert_cc(ETH_ALEN
<= sizeof(result
));
440 memcpy(mac
->ether_addr_octet
, result
, ETH_ALEN
);
442 /* see eth_random_addr in the kernel */
443 mac
->ether_addr_octet
[0] &= 0xfe; /* clear multicast bit */
444 mac
->ether_addr_octet
[0] |= 0x02; /* set local assignment bit (IEEE802) */
452 static int netdev_create(NetDev
*netdev
, Link
*link
,
453 sd_netlink_message_handler_t callback
) {
457 assert(!link
|| callback
);
460 if (NETDEV_VTABLE(netdev
)->create
) {
463 r
= NETDEV_VTABLE(netdev
)->create(netdev
);
467 log_netdev_debug(netdev
, "Created");
469 _cleanup_netlink_message_unref_ sd_netlink_message
*m
= NULL
;
471 r
= sd_rtnl_message_new_link(netdev
->manager
->rtnl
, &m
, RTM_NEWLINK
, 0);
473 return log_netdev_error_errno(netdev
, r
, "Could not allocate RTM_NEWLINK message: %m");
475 r
= sd_netlink_message_append_string(m
, IFLA_IFNAME
, netdev
->ifname
);
477 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_IFNAME, attribute: %m");
480 r
= sd_netlink_message_append_ether_addr(m
, IFLA_ADDRESS
, netdev
->mac
);
482 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_ADDRESS attribute: %m");
486 r
= sd_netlink_message_append_u32(m
, IFLA_MTU
, netdev
->mtu
);
488 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_MTU attribute: %m");
492 r
= sd_netlink_message_append_u32(m
, IFLA_LINK
, link
->ifindex
);
494 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINK attribute: %m");
497 r
= sd_netlink_message_open_container(m
, IFLA_LINKINFO
);
499 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
501 r
= sd_netlink_message_open_container_union(m
, IFLA_INFO_DATA
, netdev_kind_to_string(netdev
->kind
));
503 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
505 if (NETDEV_VTABLE(netdev
)->fill_message_create
) {
506 r
= NETDEV_VTABLE(netdev
)->fill_message_create(netdev
, link
, m
);
511 r
= sd_netlink_message_close_container(m
);
513 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
515 r
= sd_netlink_message_close_container(m
);
517 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
520 r
= sd_netlink_call_async(netdev
->manager
->rtnl
, m
, callback
, link
, 0, NULL
);
522 return log_netdev_error_errno(netdev
, r
, "Could not send rtnetlink message: %m");
526 r
= sd_netlink_call_async(netdev
->manager
->rtnl
, m
, netdev_create_handler
, netdev
, 0, NULL
);
528 return log_netdev_error_errno(netdev
, r
, "Could not send rtnetlink message: %m");
533 netdev
->state
= NETDEV_STATE_CREATING
;
535 log_netdev_debug(netdev
, "Creating");
541 /* the callback must be called, possibly after a timeout, as otherwise the Link will hang */
542 int netdev_join(NetDev
*netdev
, Link
*link
, sd_netlink_message_handler_t callback
) {
546 assert(netdev
->manager
);
547 assert(netdev
->manager
->rtnl
);
548 assert(NETDEV_VTABLE(netdev
));
550 switch (NETDEV_VTABLE(netdev
)->create_type
) {
551 case NETDEV_CREATE_MASTER
:
552 r
= netdev_enslave(netdev
, link
, callback
);
557 case NETDEV_CREATE_STACKED
:
558 r
= netdev_create(netdev
, link
, callback
);
564 assert_not_reached("Can not join independent netdev");
570 static int netdev_load_one(Manager
*manager
, const char *filename
) {
571 _cleanup_netdev_unref_ NetDev
*netdev
= NULL
;
572 _cleanup_free_ NetDev
*netdev_raw
= NULL
;
573 _cleanup_fclose_
FILE *file
= NULL
;
579 file
= fopen(filename
, "re");
587 if (null_or_empty_fd(fileno(file
))) {
588 log_debug("Skipping empty file: %s", filename
);
592 netdev_raw
= new0(NetDev
, 1);
596 netdev_raw
->kind
= _NETDEV_KIND_INVALID
;
598 r
= config_parse(NULL
, filename
, file
,
600 config_item_perf_lookup
, network_netdev_gperf_lookup
,
601 true, false, true, netdev_raw
);
605 r
= fseek(file
, 0, SEEK_SET
);
609 /* skip out early if configuration does not match the environment */
610 if (net_match_config(NULL
, NULL
, NULL
, NULL
, NULL
,
611 netdev_raw
->match_host
, netdev_raw
->match_virt
,
612 netdev_raw
->match_kernel
, netdev_raw
->match_arch
,
613 NULL
, NULL
, NULL
, NULL
, NULL
, NULL
) <= 0)
616 if (!NETDEV_VTABLE(netdev_raw
)) {
617 log_warning("NetDev with invalid Kind configured in %s. Ignoring", filename
);
621 if (!netdev_raw
->ifname
) {
622 log_warning("NetDev without Name configured in %s. Ignoring", filename
);
626 netdev
= malloc0(NETDEV_VTABLE(netdev_raw
)->object_size
);
631 netdev
->manager
= manager
;
632 netdev
->state
= _NETDEV_STATE_INVALID
;
633 netdev
->kind
= netdev_raw
->kind
;
634 netdev
->ifname
= netdev_raw
->ifname
;
636 if (NETDEV_VTABLE(netdev
)->init
)
637 NETDEV_VTABLE(netdev
)->init(netdev
);
639 r
= config_parse(NULL
, filename
, file
,
640 NETDEV_VTABLE(netdev
)->sections
,
641 config_item_perf_lookup
, network_netdev_gperf_lookup
,
642 false, false, false, netdev
);
646 /* verify configuration */
647 if (NETDEV_VTABLE(netdev
)->config_verify
) {
648 r
= NETDEV_VTABLE(netdev
)->config_verify(netdev
, filename
);
653 netdev
->filename
= strdup(filename
);
654 if (!netdev
->filename
)
658 r
= netdev_get_mac(netdev
->ifname
, &netdev
->mac
);
660 return log_error_errno(r
, "Failed to generate predictable MAC address for %s: %m", netdev
->ifname
);
663 r
= hashmap_put(netdev
->manager
->netdevs
, netdev
->ifname
, netdev
);
667 LIST_HEAD_INIT(netdev
->callbacks
);
669 log_netdev_debug(netdev
, "loaded %s", netdev_kind_to_string(netdev
->kind
));
671 switch (NETDEV_VTABLE(netdev
)->create_type
) {
672 case NETDEV_CREATE_MASTER
:
673 case NETDEV_CREATE_INDEPENDENT
:
674 r
= netdev_create(netdev
, NULL
, NULL
);
688 int netdev_load(Manager
*manager
) {
689 _cleanup_strv_free_
char **files
= NULL
;
696 while ((netdev
= hashmap_first(manager
->netdevs
)))
697 netdev_unref(netdev
);
699 r
= conf_files_list_strv(&files
, ".netdev", NULL
, network_dirs
);
701 return log_error_errno(r
, "Failed to enumerate netdev files: %m");
703 STRV_FOREACH_BACKWARDS(f
, files
) {
704 r
= netdev_load_one(manager
, *f
);