]> git.ipfire.org Git - thirdparty/systemd.git/blob - src/network/networkd-manager.c
core,logind,networkd: check for udev device initialization via enumeration matches
[thirdparty/systemd.git] / src / network / networkd-manager.c
1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
2
3 /***
4 This file is part of systemd.
5
6 Copyright 2013 Tom Gundersen <teg@jklm.no>
7
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.
12
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.
17
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/>.
20 ***/
21
22 #include "path-util.h"
23 #include "networkd.h"
24 #include "libudev-private.h"
25 #include "udev-util.h"
26
27 int manager_new(Manager **ret) {
28 _cleanup_manager_free_ Manager *m = NULL;
29 int r;
30
31 m = new0(Manager, 1);
32 if (!m)
33 return -ENOMEM;
34
35 r = sd_event_default(&m->event);
36 if (r < 0)
37 return r;
38
39 sd_event_set_watchdog(m->event, true);
40
41 r = sd_rtnl_open(RTMGRP_LINK | RTMGRP_IPV4_IFADDR, &m->rtnl);
42 if (r < 0)
43 return r;
44
45 m->udev = udev_new();
46 if (!m->udev)
47 return -ENOMEM;
48
49 m->udev_monitor = udev_monitor_new_from_netlink(m->udev, "udev");
50 if (!m->udev_monitor)
51 return -ENOMEM;
52
53 m->links = hashmap_new(uint64_hash_func, uint64_compare_func);
54 if (!m->links)
55 return -ENOMEM;
56
57 m->bridges = hashmap_new(string_hash_func, string_compare_func);
58 if (!m->bridges)
59 return -ENOMEM;
60
61 LIST_HEAD_INIT(m->networks);
62
63 m->network_dirs = strv_new("/etc/systemd/network/",
64 "/run/systemd/network/",
65 "/usr/lib/systemd/network",
66 #ifdef HAVE_SPLIT_USER
67 "/lib/systemd/network",
68 #endif
69 NULL);
70 if (!m->network_dirs)
71 return -ENOMEM;
72
73 if (!path_strv_canonicalize_uniq(m->network_dirs))
74 return -ENOMEM;
75
76 *ret = m;
77 m = NULL;
78
79 return 0;
80 }
81
82 void manager_free(Manager *m) {
83 Network *network;
84 Bridge *bridge;
85 Link *link;
86
87 udev_monitor_unref(m->udev_monitor);
88 udev_unref(m->udev);
89 sd_event_source_unref(m->udev_event_source);
90 sd_event_unref(m->event);
91
92 while ((network = m->networks))
93 network_free(network);
94
95 while ((link = hashmap_first(m->links)))
96 link_free(link);
97 hashmap_free(m->links);
98
99 while ((bridge = hashmap_first(m->bridges)))
100 bridge_free(bridge);
101 hashmap_free(m->bridges);
102
103 strv_free(m->network_dirs);
104 sd_rtnl_unref(m->rtnl);
105
106 free(m);
107 }
108
109 int manager_load_config(Manager *m) {
110 int r;
111
112 /* update timestamp */
113 paths_check_timestamp(m->network_dirs, &m->network_dirs_ts_usec, true);
114
115 r = bridge_load(m);
116 if (r < 0)
117 return r;
118
119 r = network_load(m);
120 if (r < 0)
121 return r;
122
123 return 0;
124 }
125
126 bool manager_should_reload(Manager *m) {
127 return paths_check_timestamp(m->network_dirs, &m->network_dirs_ts_usec, false);
128 }
129
130 static int manager_process_link(Manager *m, struct udev_device *device) {
131 Link *link;
132 int r;
133
134 if (streq_ptr(udev_device_get_action(device), "remove")) {
135 uint64_t ifindex;
136
137 log_debug("Link removed: %s", udev_device_get_sysname(device));
138
139 ifindex = udev_device_get_ifindex(device);
140 link = hashmap_get(m->links, &ifindex);
141 if (!link)
142 return 0;
143
144 link_free(link);
145 } else {
146 log_debug("New link: %s", udev_device_get_sysname(device));
147
148 r = link_add(m, device);
149 if (r < 0) {
150 log_error("Could not handle link %s: %s",
151 udev_device_get_sysname(device),
152 strerror(-r));
153 }
154 }
155
156 return 0;
157 }
158
159 int manager_udev_enumerate_links(Manager *m) {
160 _cleanup_udev_enumerate_unref_ struct udev_enumerate *e = NULL;
161 struct udev_list_entry *item = NULL, *first = NULL;
162 int r;
163
164 assert(m);
165
166 e = udev_enumerate_new(m->udev);
167 if (!e)
168 return -ENOMEM;
169
170 r = udev_enumerate_add_match_subsystem(e, "net");
171 if (r < 0)
172 return r;
173
174 r = udev_enumerate_add_match_is_initialized(e);
175 if (r < 0)
176 return r;
177
178 r = udev_enumerate_scan_devices(e);
179 if (r < 0)
180 return r;
181
182 first = udev_enumerate_get_list_entry(e);
183 udev_list_entry_foreach(item, first) {
184 _cleanup_udev_device_unref_ struct udev_device *d = NULL;
185 int k;
186
187 d = udev_device_new_from_syspath(m->udev, udev_list_entry_get_name(item));
188 if (!d)
189 return -ENOMEM;
190
191 k = manager_process_link(m, d);
192 if (k < 0)
193 r = k;
194 }
195
196 return r;
197 }
198
199 static int manager_dispatch_link_udev(sd_event_source *source, int fd, uint32_t revents, void *userdata) {
200 Manager *m = userdata;
201 struct udev_monitor *monitor = m->udev_monitor;
202 _cleanup_udev_device_unref_ struct udev_device *device = NULL;
203
204 device = udev_monitor_receive_device(monitor);
205 if (!device)
206 return -ENOMEM;
207
208 manager_process_link(m, device);
209 return 0;
210 }
211
212 int manager_udev_listen(Manager *m) {
213 int r;
214
215 r = udev_monitor_filter_add_match_subsystem_devtype(m->udev_monitor, "net", NULL);
216 if (r < 0) {
217 log_error("Could not add udev monitor filter: %s", strerror(-r));
218 return r;
219 }
220
221 r = udev_monitor_enable_receiving(m->udev_monitor);
222 if (r < 0) {
223 log_error("Could not enable udev monitor");
224 return r;
225 }
226
227 r = sd_event_add_io(m->event,
228 udev_monitor_get_fd(m->udev_monitor),
229 EPOLLIN, manager_dispatch_link_udev,
230 m, &m->udev_event_source);
231 if (r < 0)
232 return r;
233
234 return 0;
235 }
236
237 static int manager_rtnl_process_link(sd_rtnl *rtnl, sd_rtnl_message *message, void *userdata) {
238 Manager *m = userdata;
239 Link *link;
240 int r, ifindex;
241
242 r = sd_rtnl_message_link_get_ifindex(message, &ifindex);
243 if (r < 0)
244 return 0;
245
246 link = hashmap_get(m->links, &ifindex);
247 if (!link)
248 return 0;
249
250 r = link_update(link, message);
251 if (r < 0)
252 return 0;
253
254 return 1;
255 }
256
257 int manager_rtnl_listen(Manager *m) {
258 int r;
259
260 r = sd_rtnl_attach_event(m->rtnl, m->event, 0);
261 if (r < 0)
262 return r;
263
264 r = sd_rtnl_add_match(m->rtnl, RTM_NEWLINK, &manager_rtnl_process_link, m);
265 if (r < 0)
266 return r;
267
268 return 0;
269 }