]> git.ipfire.org Git - thirdparty/systemd.git/blame - src/network/networkd-dhcp-server.c
dhcp: list all known message type
[thirdparty/systemd.git] / src / network / networkd-dhcp-server.c
CommitLineData
db9ecf05 1/* SPDX-License-Identifier: LGPL-2.1-or-later */
8fcf1d61 2
5ae0fb7f
YW
3#include <netinet/in.h>
4#include <linux/if_arp.h>
5#include <linux/if.h>
6
8fcf1d61
YW
7#include "sd-dhcp-server.h"
8
dd1d3060
MAL
9#include "fd-util.h"
10#include "fileio.h"
093e3533 11#include "networkd-address.h"
a95e9306 12#include "networkd-dhcp-server-bus.h"
c517a49b 13#include "networkd-dhcp-server-static-lease.h"
14#include "networkd-dhcp-server.h"
8fcf1d61
YW
15#include "networkd-link.h"
16#include "networkd-manager.h"
17#include "networkd-network.h"
1d28a3cf 18#include "networkd-queue.h"
344b3cff 19#include "networkd-route-util.h"
564ca984 20#include "parse-util.h"
dd1d3060 21#include "socket-netlink.h"
564ca984
SS
22#include "string-table.h"
23#include "string-util.h"
dd1d3060 24#include "strv.h"
8fcf1d61 25
5ae0fb7f
YW
26static bool link_dhcp4_server_enabled(Link *link) {
27 assert(link);
28
29 if (link->flags & IFF_LOOPBACK)
30 return false;
31
32 if (!link->network)
33 return false;
34
5ae0fb7f
YW
35 if (link->iftype == ARPHRD_CAN)
36 return false;
37
38 return link->network->dhcp_server;
39}
40
0017ba31
YW
41void network_adjust_dhcp_server(Network *network) {
42 assert(network);
43
44 if (!network->dhcp_server)
45 return;
46
47 if (network->bond) {
48 log_warning("%s: DHCPServer= is enabled for bond slave. Disabling DHCP server.",
49 network->filename);
50 network->dhcp_server = false;
51 return;
52 }
53
54 if (!in4_addr_is_set(&network->dhcp_server_address)) {
55 Address *address;
56 bool have = false;
57
58 ORDERED_HASHMAP_FOREACH(address, network->addresses_by_section) {
59 if (section_is_invalid(address->section))
60 continue;
bab29f2a
YW
61
62 if (address->family != AF_INET)
63 continue;
64
65 if (in4_addr_is_localhost(&address->in_addr.in))
66 continue;
67
68 if (in4_addr_is_link_local(&address->in_addr.in))
69 continue;
70
71 if (in4_addr_is_set(&address->in_addr_peer.in))
72 continue;
73
74 have = true;
75 break;
0017ba31
YW
76 }
77 if (!have) {
78 log_warning("%s: DHCPServer= is enabled, but no static address configured. "
79 "Disabling DHCP server.",
80 network->filename);
81 network->dhcp_server = false;
82 return;
83 }
84 }
85}
86
3b6a3bde
YW
87int link_request_dhcp_server_address(Link *link) {
88 _cleanup_(address_freep) Address *address = NULL;
89 Address *existing;
90 int r;
91
92 assert(link);
93 assert(link->network);
94
95 if (!link_dhcp4_server_enabled(link))
96 return 0;
97
98 if (!in4_addr_is_set(&link->network->dhcp_server_address))
99 return 0;
100
101 r = address_new(&address);
102 if (r < 0)
103 return r;
104
105 address->source = NETWORK_CONFIG_SOURCE_STATIC;
106 address->family = AF_INET;
107 address->in_addr.in = link->network->dhcp_server_address;
108 address->prefixlen = link->network->dhcp_server_address_prefixlen;
e680486d 109 address_set_broadcast(address, link);
3b6a3bde
YW
110
111 if (address_get(link, address, &existing) >= 0 &&
112 address_exists(existing) &&
113 existing->source == NETWORK_CONFIG_SOURCE_STATIC)
114 /* The same address seems explicitly configured in [Address] or [Network] section.
115 * Configure the DHCP server address only when it is not. */
116 return 0;
117
118 return link_request_static_address(link, TAKE_PTR(address), true);
119}
120
0017ba31 121static int link_find_dhcp_server_address(Link *link, Address **ret) {
8fcf1d61
YW
122 Address *address;
123
124 assert(link);
125 assert(link->network);
126
0017ba31
YW
127 /* If ServerAddress= is specified, then use the address. */
128 if (in4_addr_is_set(&link->network->dhcp_server_address))
129 return link_get_ipv4_address(link, &link->network->dhcp_server_address,
130 link->network->dhcp_server_address_prefixlen, ret);
8fcf1d61 131
0017ba31 132 /* If not, then select one from static addresses. */
3b6a3bde
YW
133 SET_FOREACH(address, link->addresses) {
134 if (address->source != NETWORK_CONFIG_SOURCE_STATIC)
135 continue;
136 if (!address_exists(address))
137 continue;
138 if (address->family != AF_INET)
139 continue;
140 if (in4_addr_is_localhost(&address->in_addr.in))
141 continue;
bab29f2a
YW
142 if (in4_addr_is_link_local(&address->in_addr.in))
143 continue;
3b6a3bde
YW
144 if (in4_addr_is_set(&address->in_addr_peer.in))
145 continue;
146
147 *ret = address;
148 return 0;
149 }
8fcf1d61 150
0017ba31 151 return -ENOENT;
8fcf1d61
YW
152}
153
165d7c5c
YW
154static int dhcp_server_find_uplink(Link *link, Link **ret) {
155 assert(link);
156
157 if (link->network->dhcp_server_uplink_name)
158 return link_get_by_name(link->manager, link->network->dhcp_server_uplink_name, ret);
159
160 if (link->network->dhcp_server_uplink_index > 0)
6eab614d 161 return link_get_by_index(link->manager, link->network->dhcp_server_uplink_index, ret);
165d7c5c 162
63295b42 163 if (link->network->dhcp_server_uplink_index == UPLINK_INDEX_AUTO) {
165d7c5c
YW
164 /* It is not necessary to propagate error in automatic selection. */
165 if (manager_find_uplink(link->manager, AF_INET, link, ret) < 0)
166 *ret = NULL;
167 return 0;
168 }
169
170 *ret = NULL;
171 return 0;
172}
173
2a71d57f
LP
174static int link_push_uplink_to_dhcp_server(
175 Link *link,
2324fd3a 176 sd_dhcp_lease_server_type_t what,
2a71d57f
LP
177 sd_dhcp_server *s) {
178
8fcf1d61 179 _cleanup_free_ struct in_addr *addresses = NULL;
2a71d57f 180 bool use_dhcp_lease_data = true;
319a4f4b 181 size_t n_addresses = 0;
8fcf1d61 182
2a71d57f 183 assert(link);
8fcf1d61 184
2a71d57f
LP
185 if (!link->network)
186 return 0;
187 assert(link->network);
8fcf1d61 188
2a71d57f 189 log_link_debug(link, "Copying %s from link", dhcp_lease_server_type_to_string(what));
8fcf1d61 190
2a71d57f 191 switch (what) {
8fcf1d61 192
2a71d57f
LP
193 case SD_DHCP_LEASE_DNS:
194 /* For DNS we have a special case. We the data configured explicitly locally along with the
195 * data from the DHCP lease. */
8fcf1d61 196
2a71d57f
LP
197 for (unsigned i = 0; i < link->network->n_dns; i++) {
198 struct in_addr ia;
8fcf1d61 199
2a71d57f 200 /* Only look for IPv4 addresses */
e77bd3fd 201 if (link->network->dns[i]->family != AF_INET)
2a71d57f 202 continue;
8fcf1d61 203
e77bd3fd 204 ia = link->network->dns[i]->address.in;
2a71d57f
LP
205
206 /* Never propagate obviously borked data */
207 if (in4_addr_is_null(&ia) || in4_addr_is_localhost(&ia))
208 continue;
209
319a4f4b 210 if (!GREEDY_REALLOC(addresses, n_addresses + 1))
8fcf1d61
YW
211 return log_oom();
212
2a71d57f 213 addresses[n_addresses++] = ia;
8fcf1d61 214 }
8fcf1d61 215
2a71d57f
LP
216 use_dhcp_lease_data = link->network->dhcp_use_dns;
217 break;
8fcf1d61 218
2a71d57f
LP
219 case SD_DHCP_LEASE_NTP: {
220 char **i;
8fcf1d61 221
2a71d57f
LP
222 /* For NTP things are similar, but for NTP hostnames can be configured too, which we cannot
223 * propagate via DHCP. Hence let's only propagate those which are IP addresses. */
284e8fd0 224
2a71d57f
LP
225 STRV_FOREACH(i, link->network->ntp) {
226 union in_addr_union ia;
284e8fd0 227
2a71d57f
LP
228 if (in_addr_from_string(AF_INET, *i, &ia) < 0)
229 continue;
284e8fd0 230
2a71d57f
LP
231 /* Never propagate obviously borked data */
232 if (in4_addr_is_null(&ia.in) || in4_addr_is_localhost(&ia.in))
233 continue;
284e8fd0 234
319a4f4b 235 if (!GREEDY_REALLOC(addresses, n_addresses + 1))
2a71d57f 236 return log_oom();
284e8fd0 237
2a71d57f
LP
238 addresses[n_addresses++] = ia.in;
239 }
284e8fd0 240
2a71d57f 241 use_dhcp_lease_data = link->network->dhcp_use_ntp;
24e6f458 242 break;
2a71d57f 243 }
284e8fd0 244
ddb82ec2 245 case SD_DHCP_LEASE_SIP:
2a71d57f
LP
246
247 /* For SIP we don't allow explicit, local configuration, but there's control whether to use the data */
248 use_dhcp_lease_data = link->network->dhcp_use_sip;
24e6f458 249 break;
284e8fd0 250
2a71d57f
LP
251 case SD_DHCP_LEASE_POP3:
252 case SD_DHCP_LEASE_SMTP:
ddb82ec2 253 case SD_DHCP_LEASE_LPR:
2a71d57f
LP
254 /* For the other server types we currently do not allow local configuration of server data,
255 * since there are typically no local consumers of the data. */
c4e585a3 256 break;
d361b373 257
24e6f458 258 default:
04499a70 259 assert_not_reached();
f6269fe7
SS
260 }
261
2a71d57f 262 if (use_dhcp_lease_data && link->dhcp_lease) {
24e6f458 263 const struct in_addr *da;
f6269fe7 264
a2706075 265 int n = sd_dhcp_lease_get_servers(link->dhcp_lease, what, &da);
f6269fe7 266 if (n > 0) {
319a4f4b 267 if (!GREEDY_REALLOC(addresses, n_addresses + n))
f6269fe7
SS
268 return log_oom();
269
2a71d57f
LP
270 for (int j = 0; j < n; j++)
271 if (in4_addr_is_non_local(&da[j]))
272 addresses[n_addresses++] = da[j];
f6269fe7
SS
273 }
274 }
275
276 if (n_addresses <= 0)
277 return 0;
278
24e6f458 279 return sd_dhcp_server_set_servers(s, what, addresses, n_addresses);
299d578f
SS
280}
281
319a4f4b
LP
282static int dhcp4_server_parse_dns_server_string_and_warn(
283 const char *string,
284 struct in_addr **addresses,
285 size_t *n_addresses) {
286
dd1d3060
MAL
287 for (;;) {
288 _cleanup_free_ char *word = NULL, *server_name = NULL;
289 union in_addr_union address;
290 int family, r, ifindex = 0;
291
292 r = extract_first_word(&string, &word, NULL, 0);
293 if (r < 0)
294 return r;
295 if (r == 0)
296 break;
297
298 r = in_addr_ifindex_name_from_string_auto(word, &family, &address, &ifindex, &server_name);
299 if (r < 0) {
300 log_warning_errno(r, "Failed to parse DNS server address '%s', ignoring: %m", word);
301 continue;
302 }
303
304 /* Only look for IPv4 addresses */
305 if (family != AF_INET)
306 continue;
307
308 /* Never propagate obviously borked data */
309 if (in4_addr_is_null(&address.in) || in4_addr_is_localhost(&address.in))
310 continue;
311
319a4f4b 312 if (!GREEDY_REALLOC(*addresses, *n_addresses + 1))
dd1d3060
MAL
313 return log_oom();
314
315 (*addresses)[(*n_addresses)++] = address.in;
316 }
317
318 return 0;
319}
320
321static int dhcp4_server_set_dns_from_resolve_conf(Link *link) {
322 _cleanup_free_ struct in_addr *addresses = NULL;
dd1d3060 323 _cleanup_fclose_ FILE *f = NULL;
319a4f4b 324 size_t n_addresses = 0;
dd1d3060
MAL
325 int n = 0, r;
326
327 f = fopen(PRIVATE_UPLINK_RESOLV_CONF, "re");
328 if (!f) {
329 if (errno == ENOENT)
330 return 0;
331
332 return log_warning_errno(errno, "Failed to open " PRIVATE_UPLINK_RESOLV_CONF ": %m");
333 }
334
335 for (;;) {
336 _cleanup_free_ char *line = NULL;
337 const char *a;
338 char *l;
339
340 r = read_line(f, LONG_LINE_MAX, &line);
341 if (r < 0)
342 return log_error_errno(r, "Failed to read " PRIVATE_UPLINK_RESOLV_CONF ": %m");
343 if (r == 0)
344 break;
345
346 n++;
347
348 l = strstrip(line);
349 if (IN_SET(*l, '#', ';', 0))
350 continue;
351
352 a = first_word(l, "nameserver");
353 if (!a)
354 continue;
355
319a4f4b 356 r = dhcp4_server_parse_dns_server_string_and_warn(a, &addresses, &n_addresses);
dd1d3060
MAL
357 if (r < 0)
358 log_warning_errno(r, "Failed to parse DNS server address '%s', ignoring.", a);
359 }
360
361 if (n_addresses <= 0)
362 return 0;
363
364 return sd_dhcp_server_set_dns(link->dhcp_server, addresses, n_addresses);
365}
366
1d28a3cf 367static int dhcp4_server_configure(Link *link) {
8fcf1d61 368 bool acquired_uplink = false;
461dbb2f 369 sd_dhcp_option *p;
c517a49b 370 DHCPStaticLease *static_lease;
564ca984
SS
371 Link *uplink = NULL;
372 Address *address;
11c38d3e 373 bool bind_to_interface;
8fcf1d61
YW
374 int r;
375
5ae0fb7f
YW
376 assert(link);
377
1d28a3cf 378 log_link_debug(link, "Configuring DHCP Server.");
5ae0fb7f 379
1d28a3cf
YW
380 if (link->dhcp_server)
381 return -EBUSY;
5ae0fb7f 382
1d28a3cf
YW
383 r = sd_dhcp_server_new(&link->dhcp_server, link->ifindex);
384 if (r < 0)
385 return r;
5ae0fb7f 386
1d28a3cf
YW
387 r = sd_dhcp_server_attach_event(link->dhcp_server, link->manager->event, 0);
388 if (r < 0)
389 return r;
5ae0fb7f 390
a95e9306
LK
391 r = sd_dhcp_server_set_callback(link->dhcp_server, dhcp_server_callback, link);
392 if (r < 0)
393 return log_link_warning_errno(link, r, "Failed to set callback for DHCPv4 server instance: %m");
394
0017ba31
YW
395 r = link_find_dhcp_server_address(link, &address);
396 if (r < 0)
397 return log_link_error_errno(link, r, "Failed to find suitable address for DHCPv4 server instance: %m");
8fcf1d61
YW
398
399 /* use the server address' subnet as the pool */
400 r = sd_dhcp_server_configure_pool(link->dhcp_server, &address->in_addr.in, address->prefixlen,
401 link->network->dhcp_server_pool_offset, link->network->dhcp_server_pool_size);
402 if (r < 0)
c00c3b64 403 return log_link_error_errno(link, r, "Failed to configure address pool for DHCPv4 server instance: %m");
8fcf1d61 404
8fcf1d61
YW
405 if (link->network->dhcp_server_max_lease_time_usec > 0) {
406 r = sd_dhcp_server_set_max_lease_time(link->dhcp_server,
407 DIV_ROUND_UP(link->network->dhcp_server_max_lease_time_usec, USEC_PER_SEC));
408 if (r < 0)
c00c3b64 409 return log_link_error_errno(link, r, "Failed to set maximum lease time for DHCPv4 server instance: %m");
8fcf1d61
YW
410 }
411
412 if (link->network->dhcp_server_default_lease_time_usec > 0) {
413 r = sd_dhcp_server_set_default_lease_time(link->dhcp_server,
414 DIV_ROUND_UP(link->network->dhcp_server_default_lease_time_usec, USEC_PER_SEC));
415 if (r < 0)
c00c3b64 416 return log_link_error_errno(link, r, "Failed to set default lease time for DHCPv4 server instance: %m");
8fcf1d61
YW
417 }
418
369ac192
YW
419 r = sd_dhcp_server_set_next_server(link->dhcp_server, &link->network->dhcp_server_next_server);
420 if (r < 0)
421 return log_link_warning_errno(link, r, "Failed to set next server for DHCPv4 server instance: %m");
422
423 r = sd_dhcp_server_set_filename(link->dhcp_server, link->network->dhcp_server_filename);
424 if (r < 0)
425 return log_link_warning_errno(link, r, "Failed to set filename for DHCPv4 server instance: %m");
426
2324fd3a 427 for (sd_dhcp_lease_server_type_t type = 0; type < _SD_DHCP_LEASE_SERVER_TYPE_MAX; type ++) {
2a71d57f
LP
428
429 if (!link->network->dhcp_server_emit[type].emit)
430 continue;
431
432 if (link->network->dhcp_server_emit[type].n_addresses > 0)
433 /* Explicitly specified servers to emit */
434 r = sd_dhcp_server_set_servers(
435 link->dhcp_server,
436 type,
437 link->network->dhcp_server_emit[type].addresses,
438 link->network->dhcp_server_emit[type].n_addresses);
439 else {
440 /* Emission is requested, but nothing explicitly configured. Let's find a suitable upling */
441 if (!acquired_uplink) {
165d7c5c 442 (void) dhcp_server_find_uplink(link, &uplink);
2a71d57f
LP
443 acquired_uplink = true;
444 }
445
446 if (uplink && uplink->network)
447 r = link_push_uplink_to_dhcp_server(uplink, type, link->dhcp_server);
448 else if (type == SD_DHCP_LEASE_DNS)
449 r = dhcp4_server_set_dns_from_resolve_conf(link);
24e6f458 450 else {
2a71d57f
LP
451 log_link_debug(link,
452 "Not emitting %s on link, couldn't find suitable uplink.",
453 dhcp_lease_server_type_to_string(type));
454 continue;
24e6f458 455 }
299d578f 456 }
284e8fd0 457
2a71d57f
LP
458 if (r < 0)
459 log_link_warning_errno(link, r,
460 "Failed to set %s for DHCP server, ignoring: %m",
461 dhcp_lease_server_type_to_string(type));
462 }
463
59aa6220
YW
464 if (link->network->dhcp_server_emit_router) {
465 r = sd_dhcp_server_set_router(link->dhcp_server, &link->network->dhcp_server_router);
466 if (r < 0)
467 return log_link_error_errno(link, r, "Failed to set router address for DHCP server: %m");
468 }
8fcf1d61 469
c95df587
YA
470 r = sd_dhcp_server_set_relay_target(link->dhcp_server, &link->network->dhcp_server_relay_target);
471 if (r < 0)
472 return log_link_error_errno(link, r, "Failed to set relay target for DHCP server: %m");
473
11c38d3e
YA
474 bind_to_interface = sd_dhcp_server_is_in_relay_mode(link->dhcp_server) ? false : link->network->dhcp_server_bind_to_interface;
475 r = sd_dhcp_server_set_bind_to_interface(link->dhcp_server, bind_to_interface);
476 if (r < 0)
477 return log_link_error_errno(link, r, "Failed to set interface binding for DHCP server: %m");
478
479 r = sd_dhcp_server_set_relay_agent_information(link->dhcp_server, link->network->dhcp_server_relay_agent_circuit_id, link->network->dhcp_server_relay_agent_remote_id);
480 if (r < 0)
481 return log_link_error_errno(link, r, "Failed to set agent circuit/remote id for DHCP server: %m");
482
8fcf1d61
YW
483 if (link->network->dhcp_server_emit_timezone) {
484 _cleanup_free_ char *buffer = NULL;
7b5018ca 485 const char *tz = NULL;
8fcf1d61
YW
486
487 if (link->network->dhcp_server_timezone)
488 tz = link->network->dhcp_server_timezone;
489 else {
490 r = get_timezone(&buffer);
491 if (r < 0)
7b5018ca 492 log_link_warning_errno(link, r, "Failed to determine timezone, not sending timezone: %m");
493 else
494 tz = buffer;
8fcf1d61
YW
495 }
496
7b5018ca 497 if (tz) {
498 r = sd_dhcp_server_set_timezone(link->dhcp_server, tz);
499 if (r < 0)
500 return log_link_error_errno(link, r, "Failed to set timezone for DHCP server: %m");
501 }
8fcf1d61 502 }
564ca984 503
90e74a66 504 ORDERED_HASHMAP_FOREACH(p, link->network->dhcp_server_send_options) {
461dbb2f 505 r = sd_dhcp_server_add_option(link->dhcp_server, p);
564ca984
SS
506 if (r == -EEXIST)
507 continue;
508 if (r < 0)
c00c3b64 509 return log_link_error_errno(link, r, "Failed to set DHCPv4 option: %m");
564ca984
SS
510 }
511
90e74a66 512 ORDERED_HASHMAP_FOREACH(p, link->network->dhcp_server_send_vendor_options) {
7354900d
DW
513 r = sd_dhcp_server_add_vendor_option(link->dhcp_server, p);
514 if (r == -EEXIST)
515 continue;
516 if (r < 0)
517 return log_link_error_errno(link, r, "Failed to set DHCPv4 option: %m");
518 }
519
c517a49b 520 HASHMAP_FOREACH(static_lease, link->network->dhcp_static_leases_by_section) {
521 r = sd_dhcp_server_set_static_lease(link->dhcp_server, &static_lease->address, static_lease->client_id, static_lease->client_id_size);
522 if (r < 0)
523 return log_link_error_errno(link, r, "Failed to set DHCPv4 static lease for DHCP server: %m");
524 }
525
ab486ef4
YW
526 r = sd_dhcp_server_start(link->dhcp_server);
527 if (r < 0)
528 return log_link_error_errno(link, r, "Could not start DHCPv4 server instance: %m");
5ae0fb7f 529
ab486ef4 530 log_link_debug(link, "Offering DHCPv4 leases");
745f0620 531 return 0;
1d28a3cf
YW
532}
533
1d28a3cf 534static bool dhcp_server_is_ready_to_configure(Link *link) {
165d7c5c 535 Link *uplink = NULL;
1d28a3cf
YW
536 Address *a;
537
538 assert(link);
539
540 if (!link->network)
541 return false;
542
543 if (!IN_SET(link->state, LINK_STATE_CONFIGURING, LINK_STATE_CONFIGURED))
544 return false;
545
baa95d22
YW
546 if (link->set_flags_messages > 0)
547 return false;
548
1d28a3cf
YW
549 if (!link_has_carrier(link))
550 return false;
551
1d28a3cf
YW
552 if (!link->static_addresses_configured)
553 return false;
554
555 if (link_find_dhcp_server_address(link, &a) < 0)
556 return false;
557
558 if (!address_is_ready(a))
559 return false;
560
165d7c5c
YW
561 if (dhcp_server_find_uplink(link, &uplink) < 0)
562 return false;
563
564 if (uplink && !uplink->network)
565 return false;
566
1d28a3cf
YW
567 return true;
568}
569
570int request_process_dhcp_server(Request *req) {
745f0620
YW
571 Link *link;
572 int r;
573
1d28a3cf 574 assert(req);
1d28a3cf 575 assert(req->type == REQUEST_TYPE_DHCP_SERVER);
745f0620 576 assert_se(link = req->link);
1d28a3cf 577
745f0620 578 if (!dhcp_server_is_ready_to_configure(link))
1d28a3cf
YW
579 return 0;
580
745f0620
YW
581 r = dhcp4_server_configure(link);
582 if (r < 0)
583 return log_link_warning_errno(link, r, "Failed to configure DHCP server: %m");
584
585 return 1;
8fcf1d61
YW
586}
587
8bed7c55 588int link_request_dhcp_server(Link *link) {
745f0620
YW
589 int r;
590
8bed7c55
YW
591 assert(link);
592
593 if (!link_dhcp4_server_enabled(link))
594 return 0;
595
596 if (link->dhcp_server)
597 return 0;
598
599 log_link_debug(link, "Requesting DHCP server.");
745f0620
YW
600 r = link_queue_request(link, REQUEST_TYPE_DHCP_SERVER, NULL, false, NULL, NULL, NULL);
601 if (r < 0)
602 return log_link_warning_errno(link, r, "Failed to request configuration of DHCP server: %m");
603
604 return 0;
8bed7c55
YW
605}
606
11c38d3e
YA
607int config_parse_dhcp_server_relay_agent_suboption(
608 const char *unit,
609 const char *filename,
610 unsigned line,
611 const char *section,
612 unsigned section_line,
613 const char *lvalue,
614 int ltype,
615 const char *rvalue,
616 void *data,
617 void *userdata) {
618
619 char **suboption_value = data;
620 char* p;
621
622 assert(filename);
623 assert(lvalue);
624 assert(rvalue);
625
11c38d3e
YA
626 if (isempty(rvalue)) {
627 *suboption_value = mfree(*suboption_value);
628 return 0;
629 }
630
631 p = startswith(rvalue, "string:");
632 if (!p) {
633 log_syntax(unit, LOG_WARNING, filename, line, 0,
634 "Failed to parse %s=%s'. Invalid format, ignoring.", lvalue, rvalue);
635 return 0;
636 }
637 return free_and_strdup(suboption_value, empty_to_null(p));
638}
639
2a71d57f 640int config_parse_dhcp_server_emit(
8fcf1d61
YW
641 const char *unit,
642 const char *filename,
643 unsigned line,
2a71d57f
LP
644 const char *section,
645 unsigned section_line,
8fcf1d61 646 const char *lvalue,
2a71d57f 647 int ltype,
8fcf1d61 648 const char *rvalue,
2a71d57f
LP
649 void *data,
650 void *userdata) {
8fcf1d61 651
2a71d57f
LP
652 NetworkDHCPServerEmitAddress *emit = data;
653
654 assert(emit);
8fcf1d61
YW
655 assert(rvalue);
656
faa1b3c6
YW
657 if (isempty(rvalue)) {
658 emit->addresses = mfree(emit->addresses);
659 emit->n_addresses = 0;
660 return 0;
661 }
662
c1997a5b 663 for (const char *p = rvalue;;) {
8fcf1d61
YW
664 _cleanup_free_ char *w = NULL;
665 union in_addr_union a;
c1997a5b 666 int r;
8fcf1d61
YW
667
668 r = extract_first_word(&p, &w, NULL, 0);
669 if (r == -ENOMEM)
670 return log_oom();
671 if (r < 0) {
d96edb2c 672 log_syntax(unit, LOG_WARNING, filename, line, r,
8fcf1d61
YW
673 "Failed to extract word, ignoring: %s", rvalue);
674 return 0;
675 }
676 if (r == 0)
c1997a5b 677 return 0;
8fcf1d61 678
5f468b9f
YW
679 if (streq(w, "_server_address"))
680 a = IN_ADDR_NULL; /* null address will be converted to the server address. */
681 else {
682 r = in_addr_from_string(AF_INET, w, &a);
683 if (r < 0) {
684 log_syntax(unit, LOG_WARNING, filename, line, r,
685 "Failed to parse %s= address '%s', ignoring: %m", lvalue, w);
686 continue;
687 }
688
689 if (in4_addr_is_null(&a.in)) {
690 log_syntax(unit, LOG_WARNING, filename, line, 0,
691 "Found a null address in %s=, ignoring.", lvalue);
692 continue;
693 }
8fcf1d61
YW
694 }
695
77e73102 696 if (!GREEDY_REALLOC(emit->addresses, emit->n_addresses + 1))
8fcf1d61
YW
697 return log_oom();
698
2a71d57f 699 emit->addresses[emit->n_addresses++] = a.in;
8fcf1d61 700 }
8fcf1d61 701}
0017ba31
YW
702
703int config_parse_dhcp_server_address(
704 const char *unit,
705 const char *filename,
706 unsigned line,
707 const char *section,
708 unsigned section_line,
709 const char *lvalue,
710 int ltype,
711 const char *rvalue,
712 void *data,
713 void *userdata) {
714
715 Network *network = userdata;
716 union in_addr_union a;
717 unsigned char prefixlen;
718 int r;
719
720 assert(filename);
721 assert(lvalue);
722 assert(rvalue);
723
724 if (isempty(rvalue)) {
725 network->dhcp_server_address = (struct in_addr) {};
726 network->dhcp_server_address_prefixlen = 0;
727 return 0;
728 }
729
730 r = in_addr_prefix_from_string(rvalue, AF_INET, &a, &prefixlen);
731 if (r < 0) {
732 log_syntax(unit, LOG_WARNING, filename, line, r,
733 "Failed to parse %s=, ignoring assignment: %s", lvalue, rvalue);
734 return 0;
735 }
736 if (in4_addr_is_null(&a.in) || in4_addr_is_localhost(&a.in)) {
737 log_syntax(unit, LOG_WARNING, filename, line, 0,
738 "DHCP server address cannot be the ANY address or a localhost address, "
739 "ignoring assignment: %s", rvalue);
740 return 0;
741 }
742
743 network->dhcp_server_address = a.in;
744 network->dhcp_server_address_prefixlen = prefixlen;
745 return 0;
746}