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 "siphash24.h"
28 #include "netlink-util.h"
29 #include "network-internal.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
);
251 /* callback for netdev's created without a backing Link */
252 static int netdev_create_handler(sd_netlink
*rtnl
, sd_netlink_message
*m
, void *userdata
) {
253 _cleanup_netdev_unref_ NetDev
*netdev
= userdata
;
256 assert(netdev
->state
!= _NETDEV_STATE_INVALID
);
258 r
= sd_netlink_message_get_errno(m
);
260 log_netdev_info(netdev
, "netdev exists, using existing without changing its parameters");
262 log_netdev_warning_errno(netdev
, r
, "netdev could not be created: %m");
268 log_netdev_debug(netdev
, "Created");
273 int netdev_enslave(NetDev
*netdev
, Link
*link
, sd_netlink_message_handler_t callback
) {
277 assert(netdev
->manager
);
278 assert(netdev
->manager
->rtnl
);
279 assert(IN_SET(netdev
->kind
, NETDEV_KIND_BRIDGE
, NETDEV_KIND_BOND
));
281 if (netdev
->state
== NETDEV_STATE_READY
) {
282 r
= netdev_enslave_ready(netdev
, link
, callback
);
285 } else if (IN_SET(netdev
->state
, NETDEV_STATE_LINGER
, NETDEV_STATE_FAILED
)) {
286 _cleanup_netlink_message_unref_ sd_netlink_message
*m
= NULL
;
288 r
= rtnl_message_new_synthetic_error(-ENODEV
, 0, &m
);
290 callback(netdev
->manager
->rtnl
, m
, link
);
292 /* the netdev is not yet read, save this request for when it is */
293 netdev_join_callback
*cb
;
295 cb
= new0(netdev_join_callback
, 1);
299 cb
->callback
= callback
;
303 LIST_PREPEND(callbacks
, netdev
->callbacks
, cb
);
305 log_netdev_debug(netdev
, "Will enslave '%s', when ready", link
->ifname
);
311 int netdev_set_ifindex(NetDev
*netdev
, sd_netlink_message
*message
) {
314 const char *received_kind
;
315 const char *received_name
;
321 r
= sd_netlink_message_get_type(message
, &type
);
323 return log_netdev_error_errno(netdev
, r
, "Could not get rtnl message type: %m");
325 if (type
!= RTM_NEWLINK
) {
326 log_netdev_error(netdev
, "Cannot set ifindex from unexpected rtnl message type.");
330 r
= sd_rtnl_message_link_get_ifindex(message
, &ifindex
);
332 log_netdev_error_errno(netdev
, r
, "Could not get ifindex: %m");
333 netdev_enter_failed(netdev
);
335 } else if (ifindex
<= 0) {
336 log_netdev_error(netdev
, "Got invalid ifindex: %d", ifindex
);
337 netdev_enter_failed(netdev
);
341 if (netdev
->ifindex
> 0) {
342 if (netdev
->ifindex
!= ifindex
) {
343 log_netdev_error(netdev
, "Could not set ifindex to %d, already set to %d",
344 ifindex
, netdev
->ifindex
);
345 netdev_enter_failed(netdev
);
348 /* ifindex already set to the same for this netdev */
352 r
= sd_netlink_message_read_string(message
, IFLA_IFNAME
, &received_name
);
354 return log_netdev_error_errno(netdev
, r
, "Could not get IFNAME: %m");
356 if (!streq(netdev
->ifname
, received_name
)) {
357 log_netdev_error(netdev
, "Received newlink with wrong IFNAME %s", received_name
);
358 netdev_enter_failed(netdev
);
362 r
= sd_netlink_message_enter_container(message
, IFLA_LINKINFO
);
364 return log_netdev_error_errno(netdev
, r
, "Could not get LINKINFO: %m");
366 r
= sd_netlink_message_read_string(message
, IFLA_INFO_KIND
, &received_kind
);
368 return log_netdev_error_errno(netdev
, r
, "Could not get KIND: %m");
370 r
= sd_netlink_message_exit_container(message
);
372 return log_netdev_error_errno(netdev
, r
, "Could not exit container: %m");
374 if (netdev
->kind
== NETDEV_KIND_TAP
)
375 /* the kernel does not distinguish between tun and tap */
378 kind
= netdev_kind_to_string(netdev
->kind
);
380 log_netdev_error(netdev
, "Could not get kind");
381 netdev_enter_failed(netdev
);
386 if (!streq(kind
, received_kind
)) {
387 log_netdev_error(netdev
,
388 "Received newlink with wrong KIND %s, "
389 "expected %s", received_kind
, kind
);
390 netdev_enter_failed(netdev
);
394 netdev
->ifindex
= ifindex
;
396 log_netdev_debug(netdev
, "netdev has index %d", netdev
->ifindex
);
398 netdev_enter_ready(netdev
);
403 #define HASH_KEY SD_ID128_MAKE(52,e1,45,bd,00,6f,29,96,21,c6,30,6d,83,71,04,48)
405 int netdev_get_mac(const char *ifname
, struct ether_addr
**ret
) {
406 _cleanup_free_
struct ether_addr
*mac
= NULL
;
415 mac
= new0(struct ether_addr
, 1);
420 sz
= sizeof(sd_id128_t
) + l
;
423 /* fetch some persistent data unique to the machine */
424 r
= sd_id128_get_machine((sd_id128_t
*) v
);
428 /* combine with some data unique (on this machine) to this
430 memcpy(v
+ sizeof(sd_id128_t
), ifname
, l
);
432 /* Let's hash the host machine ID plus the container name. We
433 * use a fixed, but originally randomly created hash key here. */
434 siphash24(result
, v
, sz
, HASH_KEY
.bytes
);
436 assert_cc(ETH_ALEN
<= sizeof(result
));
437 memcpy(mac
->ether_addr_octet
, result
, ETH_ALEN
);
439 /* see eth_random_addr in the kernel */
440 mac
->ether_addr_octet
[0] &= 0xfe; /* clear multicast bit */
441 mac
->ether_addr_octet
[0] |= 0x02; /* set local assignment bit (IEEE802) */
449 static int netdev_create(NetDev
*netdev
, Link
*link
,
450 sd_netlink_message_handler_t callback
) {
454 assert(!link
|| callback
);
457 if (NETDEV_VTABLE(netdev
)->create
) {
460 r
= NETDEV_VTABLE(netdev
)->create(netdev
);
464 log_netdev_debug(netdev
, "Created");
466 _cleanup_netlink_message_unref_ sd_netlink_message
*m
= NULL
;
468 r
= sd_rtnl_message_new_link(netdev
->manager
->rtnl
, &m
, RTM_NEWLINK
, 0);
470 return log_netdev_error_errno(netdev
, r
, "Could not allocate RTM_NEWLINK message: %m");
472 r
= sd_netlink_message_append_string(m
, IFLA_IFNAME
, netdev
->ifname
);
474 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_IFNAME, attribute: %m");
477 r
= sd_netlink_message_append_ether_addr(m
, IFLA_ADDRESS
, netdev
->mac
);
479 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_ADDRESS attribute: %m");
483 r
= sd_netlink_message_append_u32(m
, IFLA_MTU
, netdev
->mtu
);
485 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_MTU attribute: %m");
489 r
= sd_netlink_message_append_u32(m
, IFLA_LINK
, link
->ifindex
);
491 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINK attribute: %m");
494 r
= sd_netlink_message_open_container(m
, IFLA_LINKINFO
);
496 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
498 r
= sd_netlink_message_open_container_union(m
, IFLA_INFO_DATA
, netdev_kind_to_string(netdev
->kind
));
500 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_INFO_DATA attribute: %m");
502 if (NETDEV_VTABLE(netdev
)->fill_message_create
) {
503 r
= NETDEV_VTABLE(netdev
)->fill_message_create(netdev
, link
, m
);
508 r
= sd_netlink_message_close_container(m
);
510 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
512 r
= sd_netlink_message_close_container(m
);
514 return log_netdev_error_errno(netdev
, r
, "Could not append IFLA_LINKINFO attribute: %m");
517 r
= sd_netlink_call_async(netdev
->manager
->rtnl
, m
, callback
, link
, 0, NULL
);
519 return log_netdev_error_errno(netdev
, r
, "Could not send rtnetlink message: %m");
523 r
= sd_netlink_call_async(netdev
->manager
->rtnl
, m
, netdev_create_handler
, netdev
, 0, NULL
);
525 return log_netdev_error_errno(netdev
, r
, "Could not send rtnetlink message: %m");
530 netdev
->state
= NETDEV_STATE_CREATING
;
532 log_netdev_debug(netdev
, "Creating");
538 /* the callback must be called, possibly after a timeout, as otherwise the Link will hang */
539 int netdev_join(NetDev
*netdev
, Link
*link
, sd_netlink_message_handler_t callback
) {
543 assert(netdev
->manager
);
544 assert(netdev
->manager
->rtnl
);
545 assert(NETDEV_VTABLE(netdev
));
547 switch (NETDEV_VTABLE(netdev
)->create_type
) {
548 case NETDEV_CREATE_MASTER
:
549 r
= netdev_enslave(netdev
, link
, callback
);
554 case NETDEV_CREATE_STACKED
:
555 r
= netdev_create(netdev
, link
, callback
);
561 assert_not_reached("Can not join independent netdev");
567 static int netdev_load_one(Manager
*manager
, const char *filename
) {
568 _cleanup_netdev_unref_ NetDev
*netdev
= NULL
;
569 _cleanup_free_ NetDev
*netdev_raw
= NULL
;
570 _cleanup_fclose_
FILE *file
= NULL
;
576 file
= fopen(filename
, "re");
584 if (null_or_empty_fd(fileno(file
))) {
585 log_debug("Skipping empty file: %s", filename
);
589 netdev_raw
= new0(NetDev
, 1);
593 netdev_raw
->kind
= _NETDEV_KIND_INVALID
;
595 r
= config_parse(NULL
, filename
, file
,
597 config_item_perf_lookup
, network_netdev_gperf_lookup
,
598 true, false, true, netdev_raw
);
602 r
= fseek(file
, 0, SEEK_SET
);
606 /* skip out early if configuration does not match the environment */
607 if (net_match_config(NULL
, NULL
, NULL
, NULL
, NULL
,
608 netdev_raw
->match_host
, netdev_raw
->match_virt
,
609 netdev_raw
->match_kernel
, netdev_raw
->match_arch
,
610 NULL
, NULL
, NULL
, NULL
, NULL
, NULL
) <= 0)
613 if (!NETDEV_VTABLE(netdev_raw
)) {
614 log_warning("NetDev with invalid Kind configured in %s. Ignoring", filename
);
618 if (!netdev_raw
->ifname
) {
619 log_warning("NetDev without Name configured in %s. Ignoring", filename
);
623 netdev
= malloc0(NETDEV_VTABLE(netdev_raw
)->object_size
);
628 netdev
->manager
= manager
;
629 netdev
->state
= _NETDEV_STATE_INVALID
;
630 netdev
->kind
= netdev_raw
->kind
;
631 netdev
->ifname
= netdev_raw
->ifname
;
633 if (NETDEV_VTABLE(netdev
)->init
)
634 NETDEV_VTABLE(netdev
)->init(netdev
);
636 r
= config_parse(NULL
, filename
, file
,
637 NETDEV_VTABLE(netdev
)->sections
,
638 config_item_perf_lookup
, network_netdev_gperf_lookup
,
639 false, false, false, netdev
);
643 /* verify configuration */
644 if (NETDEV_VTABLE(netdev
)->config_verify
) {
645 r
= NETDEV_VTABLE(netdev
)->config_verify(netdev
, filename
);
650 netdev
->filename
= strdup(filename
);
651 if (!netdev
->filename
)
655 r
= netdev_get_mac(netdev
->ifname
, &netdev
->mac
);
657 return log_error_errno(r
, "Failed to generate predictable MAC address for %s: %m", netdev
->ifname
);
660 r
= hashmap_put(netdev
->manager
->netdevs
, netdev
->ifname
, netdev
);
664 LIST_HEAD_INIT(netdev
->callbacks
);
666 log_netdev_debug(netdev
, "loaded %s", netdev_kind_to_string(netdev
->kind
));
668 switch (NETDEV_VTABLE(netdev
)->create_type
) {
669 case NETDEV_CREATE_MASTER
:
670 case NETDEV_CREATE_INDEPENDENT
:
671 r
= netdev_create(netdev
, NULL
, NULL
);
685 int netdev_load(Manager
*manager
) {
686 _cleanup_strv_free_
char **files
= NULL
;
693 while ((netdev
= hashmap_first(manager
->netdevs
)))
694 netdev_unref(netdev
);
696 r
= conf_files_list_strv(&files
, ".netdev", NULL
, network_dirs
);
698 return log_error_errno(r
, "Failed to enumerate netdev files: %m");
700 STRV_FOREACH_BACKWARDS(f
, files
) {
701 r
= netdev_load_one(manager
, *f
);