chiark / gitweb /
networkd: netdev - split out bridge creation
[elogind.git] / src / network / networkd-bridge.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3 /***
4     This file is part of systemd.
5
6     Copyright 2014  Tom Gundersen <teg@jklm.no>
7     Copyright 2014  Susant Sahani
8
9     systemd is free software; you can redistribute it and/or modify it
10     under the terms of the GNU Lesser General Public License as published by
11     the Free Software Foundation; either version 2.1 of the License, or
12     (at your option) any later version.
13
14     systemd is distributed in the hope that it will be useful, but
15     WITHOUT ANY WARRANTY; without even the implied warranty of
16     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17     Lesser General Public License for more details.
18
19     You should have received a copy of the GNU Lesser General Public License
20     along with systemd; If not, see <http://www.gnu.org/licenses/>.
21 ***/
22
23 #include <netinet/ether.h>
24 #include <arpa/inet.h>
25 #include <net/if.h>
26
27 #include "sd-rtnl.h"
28 #include "networkd.h"
29 #include "missing.h"
30
31 static int netdev_fill_bridge_rtnl_message(NetDev *netdev, sd_rtnl_message *m) {
32         int r;
33
34         assert(m);
35
36         r = sd_rtnl_message_append_string(m, IFLA_IFNAME, netdev->ifname);
37         if (r < 0) {
38                 log_error_netdev(netdev,
39                                  "Could not append IFLA_IFNAME, attribute: %s",
40                                  strerror(-r));
41                 return r;
42         }
43
44         r = sd_rtnl_message_open_container(m, IFLA_LINKINFO);
45         if (r < 0) {
46                 log_error_netdev(netdev,
47                                  "Could not append IFLA_LINKINFO attribute: %s",
48                                  strerror(-r));
49                 return r;
50         }
51
52         r = sd_rtnl_message_open_container_union(m, IFLA_INFO_DATA,
53                                                  netdev_kind_to_string(netdev->kind));
54         if (r < 0) {
55                 log_error_netdev(netdev,
56                                  "Could not append IFLA_INFO_DATA attribute: %s",
57                                  strerror(-r));
58                 return r;
59         }
60
61         r = sd_rtnl_message_close_container(m);
62         if (r < 0) {
63                 log_error_netdev(netdev,
64                                  "Could not append IFLA_LINKINFO attribute: %s",
65                                  strerror(-r));
66                 return r;
67         }
68
69         r = sd_rtnl_message_close_container(m);
70         if (r < 0) {
71                 log_error_netdev(netdev,
72                                  "Could not append IFLA_LINKINFO attribute: %s",
73                                  strerror(-r));
74                 return r;
75         }
76
77         return r;
78 }
79
80 int netdev_create_bridge(NetDev *netdev, sd_rtnl_message_handler_t callback) {
81         _cleanup_rtnl_message_unref_ sd_rtnl_message *m = NULL;
82         int r;
83
84         assert(netdev);
85         assert(netdev->kind == NETDEV_KIND_BRIDGE);
86         assert(netdev->ifname);
87         assert(netdev->manager);
88         assert(netdev->manager->rtnl);
89
90         r = sd_rtnl_message_new_link(netdev->manager->rtnl, &m, RTM_NEWLINK, 0);
91         if (r < 0) {
92                 log_error_netdev(netdev,
93                                  "Could not allocate RTM_NEWLINK message: %s",
94                                  strerror(-r));
95                 return r;
96         }
97
98         r = netdev_fill_bridge_rtnl_message(netdev, m);
99         if(r < 0)
100                 return r;
101
102         r = sd_rtnl_call_async(netdev->manager->rtnl, m, callback, netdev, 0, NULL);
103         if (r < 0) {
104                 log_error_netdev(netdev,
105                                  "Could not send rtnetlink message: %s", strerror(-r));
106                 return r;
107         }
108
109         netdev_ref(netdev);
110
111         log_debug_netdev(netdev, "Creating bridge netdev.");
112
113         netdev->state = NETDEV_STATE_CREATING;
114
115         return 0;
116 }