2017-11-18 17:09:20 +01:00
|
|
|
/* SPDX-License-Identifier: LGPL-2.1+ */
|
2013-10-28 20:59:56 +01:00
|
|
|
/***
|
|
|
|
This file is part of systemd.
|
|
|
|
|
|
|
|
Copyright (C) 2013 Tom Gundersen <teg@jklm.no>
|
|
|
|
|
|
|
|
systemd is free software; you can redistribute it and/or modify it
|
|
|
|
under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
systemd is distributed in the hope that it will be useful, but
|
|
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
|
|
along with systemd; If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
***/
|
|
|
|
|
2013-11-09 22:19:42 +01:00
|
|
|
#include <arpa/inet.h>
|
2015-10-24 22:58:24 +02:00
|
|
|
#include <linux/if.h>
|
|
|
|
#include <netinet/ether.h>
|
2013-10-28 20:59:56 +01:00
|
|
|
|
2018-01-11 00:39:12 +01:00
|
|
|
#include "sd-id128.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "sd-ndisc.h"
|
|
|
|
|
2015-10-27 03:01:06 +01:00
|
|
|
#include "alloc-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "condition.h"
|
|
|
|
#include "conf-parser.h"
|
2014-06-28 00:00:06 +02:00
|
|
|
#include "dhcp-lease-internal.h"
|
2016-05-31 13:00:54 +02:00
|
|
|
#include "ether-addr-util.h"
|
2015-11-16 22:09:36 +01:00
|
|
|
#include "hexdecoct.h"
|
2013-11-02 02:13:48 +01:00
|
|
|
#include "log.h"
|
2015-10-26 16:18:16 +01:00
|
|
|
#include "network-internal.h"
|
|
|
|
#include "parse-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "siphash24.h"
|
2016-05-06 21:20:59 +02:00
|
|
|
#include "socket-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "string-util.h"
|
|
|
|
#include "strv.h"
|
2013-10-28 20:59:56 +01:00
|
|
|
#include "utf8.h"
|
2013-10-29 15:59:45 +01:00
|
|
|
#include "util.h"
|
2013-10-28 20:59:56 +01:00
|
|
|
|
2014-06-19 14:39:05 +02:00
|
|
|
const char *net_get_name(struct udev_device *device) {
|
2014-08-11 22:44:51 +02:00
|
|
|
const char *name, *field;
|
2014-06-19 14:39:05 +02:00
|
|
|
|
|
|
|
assert(device);
|
2014-03-21 19:23:35 +01:00
|
|
|
|
|
|
|
/* fetch some persistent data unique (on this machine) to this device */
|
2014-08-11 22:44:51 +02:00
|
|
|
FOREACH_STRING(field, "ID_NET_NAME_ONBOARD", "ID_NET_NAME_SLOT", "ID_NET_NAME_PATH", "ID_NET_NAME_MAC") {
|
2014-03-21 19:23:35 +01:00
|
|
|
name = udev_device_get_property_value(device, field);
|
|
|
|
if (name)
|
2014-08-11 22:44:51 +02:00
|
|
|
return name;
|
2014-03-21 19:23:35 +01:00
|
|
|
}
|
|
|
|
|
2014-08-11 22:44:51 +02:00
|
|
|
return NULL;
|
2014-06-19 14:39:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#define HASH_KEY SD_ID128_MAKE(d3,1e,48,fa,90,fe,4b,4c,9d,af,d5,d7,a1,b1,2e,8a)
|
|
|
|
|
2015-11-16 09:21:20 +01:00
|
|
|
int net_get_unique_predictable_data(struct udev_device *device, uint64_t *result) {
|
2014-06-19 14:39:05 +02:00
|
|
|
size_t l, sz = 0;
|
|
|
|
const char *name = NULL;
|
|
|
|
int r;
|
|
|
|
uint8_t *v;
|
|
|
|
|
|
|
|
assert(device);
|
|
|
|
|
|
|
|
name = net_get_name(device);
|
2014-03-21 19:23:35 +01:00
|
|
|
if (!name)
|
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
l = strlen(name);
|
|
|
|
sz = sizeof(sd_id128_t) + l;
|
|
|
|
v = alloca(sz);
|
|
|
|
|
|
|
|
/* fetch some persistent data unique to this machine */
|
|
|
|
r = sd_id128_get_machine((sd_id128_t*) v);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
memcpy(v + sizeof(sd_id128_t), name, l);
|
|
|
|
|
|
|
|
/* Let's hash the machine ID plus the device name. We
|
|
|
|
* use a fixed, but originally randomly created hash
|
|
|
|
* key here. */
|
2015-11-16 23:17:52 +01:00
|
|
|
*result = htole64(siphash24(v, sz, HASH_KEY.bytes));
|
2014-03-21 19:23:35 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-12-07 22:42:17 +01:00
|
|
|
static bool net_condition_test_strv(char * const *raw_patterns,
|
|
|
|
const char *string) {
|
2016-12-07 19:12:10 +01:00
|
|
|
if (strv_isempty(raw_patterns))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
/* If the patterns begin with "!", edit it out and negate the test. */
|
|
|
|
if (raw_patterns[0][0] == '!') {
|
|
|
|
char **patterns;
|
|
|
|
unsigned i, length;
|
|
|
|
|
|
|
|
length = strv_length(raw_patterns) + 1; /* Include the NULL. */
|
|
|
|
patterns = newa(char*, length);
|
|
|
|
patterns[0] = raw_patterns[0] + 1; /* Skip the "!". */
|
|
|
|
for (i = 1; i < length; i++)
|
|
|
|
patterns[i] = raw_patterns[i];
|
|
|
|
|
|
|
|
return !string || !strv_fnmatch(patterns, string, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return string && strv_fnmatch(raw_patterns, string, 0);
|
|
|
|
}
|
|
|
|
|
2013-11-02 02:13:48 +01:00
|
|
|
bool net_match_config(const struct ether_addr *match_mac,
|
2015-02-10 18:30:16 +01:00
|
|
|
char * const *match_paths,
|
|
|
|
char * const *match_drivers,
|
|
|
|
char * const *match_types,
|
|
|
|
char * const *match_names,
|
2014-02-20 19:39:49 +01:00
|
|
|
Condition *match_host,
|
|
|
|
Condition *match_virt,
|
core,udev,networkd: add ConditionKernelVersion=
This adds a simple condition/assert/match to the service manager, to
udev's .link handling and to networkd, for matching the kernel version
string.
In this version we only do fnmatch() based globbing, but we might want
to extend that to version comparisons later on, if we like, by slightly
extending the syntax with ">=", "<=", ">", "<" and "==" expressions.
2017-12-13 20:34:13 +01:00
|
|
|
Condition *match_kernel_cmdline,
|
|
|
|
Condition *match_kernel_version,
|
2014-02-21 14:51:19 +01:00
|
|
|
Condition *match_arch,
|
2014-04-15 14:21:44 +02:00
|
|
|
const struct ether_addr *dev_mac,
|
2013-11-05 01:35:26 +01:00
|
|
|
const char *dev_path,
|
2014-02-21 22:29:25 +01:00
|
|
|
const char *dev_parent_driver,
|
2013-11-05 01:35:26 +01:00
|
|
|
const char *dev_driver,
|
|
|
|
const char *dev_type,
|
2014-12-05 15:56:10 +01:00
|
|
|
const char *dev_name) {
|
2013-11-02 02:13:48 +01:00
|
|
|
|
2016-07-01 00:56:23 +02:00
|
|
|
if (match_host && condition_test(match_host) <= 0)
|
2014-12-04 18:12:55 +01:00
|
|
|
return false;
|
2014-02-20 19:39:49 +01:00
|
|
|
|
2016-07-01 00:56:23 +02:00
|
|
|
if (match_virt && condition_test(match_virt) <= 0)
|
2014-12-04 18:12:55 +01:00
|
|
|
return false;
|
2014-02-20 19:39:49 +01:00
|
|
|
|
core,udev,networkd: add ConditionKernelVersion=
This adds a simple condition/assert/match to the service manager, to
udev's .link handling and to networkd, for matching the kernel version
string.
In this version we only do fnmatch() based globbing, but we might want
to extend that to version comparisons later on, if we like, by slightly
extending the syntax with ">=", "<=", ">", "<" and "==" expressions.
2017-12-13 20:34:13 +01:00
|
|
|
if (match_kernel_cmdline && condition_test(match_kernel_cmdline) <= 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (match_kernel_version && condition_test(match_kernel_version) <= 0)
|
2014-12-04 18:12:55 +01:00
|
|
|
return false;
|
2014-02-20 19:39:49 +01:00
|
|
|
|
2016-07-01 00:56:23 +02:00
|
|
|
if (match_arch && condition_test(match_arch) <= 0)
|
2014-12-04 18:12:55 +01:00
|
|
|
return false;
|
2014-02-21 14:51:19 +01:00
|
|
|
|
2014-04-15 14:21:44 +02:00
|
|
|
if (match_mac && (!dev_mac || memcmp(match_mac, dev_mac, ETH_ALEN)))
|
2014-12-04 18:12:55 +01:00
|
|
|
return false;
|
2013-11-02 02:13:48 +01:00
|
|
|
|
2016-12-07 19:12:10 +01:00
|
|
|
if (!net_condition_test_strv(match_paths, dev_path))
|
2015-02-14 00:38:22 +01:00
|
|
|
return false;
|
2015-02-10 18:30:16 +01:00
|
|
|
|
2016-12-07 19:12:10 +01:00
|
|
|
if (!net_condition_test_strv(match_drivers, dev_driver))
|
2015-02-14 00:38:22 +01:00
|
|
|
return false;
|
2015-02-10 18:30:16 +01:00
|
|
|
|
2016-12-07 19:12:10 +01:00
|
|
|
if (!net_condition_test_strv(match_types, dev_type))
|
2015-02-14 00:38:22 +01:00
|
|
|
return false;
|
2015-02-10 18:30:16 +01:00
|
|
|
|
2016-12-07 19:12:10 +01:00
|
|
|
if (!net_condition_test_strv(match_names, dev_name))
|
2015-02-14 00:38:22 +01:00
|
|
|
return false;
|
2015-02-10 18:30:16 +01:00
|
|
|
|
2014-12-04 18:12:55 +01:00
|
|
|
return true;
|
2013-11-02 02:13:48 +01:00
|
|
|
}
|
2013-10-28 20:59:56 +01:00
|
|
|
|
2014-02-20 19:39:49 +01:00
|
|
|
int config_parse_net_condition(const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
unsigned line,
|
|
|
|
const char *section,
|
|
|
|
unsigned section_line,
|
|
|
|
const char *lvalue,
|
|
|
|
int ltype,
|
|
|
|
const char *rvalue,
|
|
|
|
void *data,
|
|
|
|
void *userdata) {
|
|
|
|
|
|
|
|
ConditionType cond = ltype;
|
|
|
|
Condition **ret = data;
|
|
|
|
bool negate;
|
|
|
|
Condition *c;
|
|
|
|
_cleanup_free_ char *s = NULL;
|
|
|
|
|
|
|
|
assert(filename);
|
|
|
|
assert(lvalue);
|
|
|
|
assert(rvalue);
|
|
|
|
assert(data);
|
|
|
|
|
|
|
|
negate = rvalue[0] == '!';
|
|
|
|
if (negate)
|
|
|
|
rvalue++;
|
|
|
|
|
|
|
|
s = strdup(rvalue);
|
|
|
|
if (!s)
|
|
|
|
return log_oom();
|
|
|
|
|
|
|
|
c = condition_new(cond, s, false, negate);
|
|
|
|
if (!c)
|
|
|
|
return log_oom();
|
|
|
|
|
|
|
|
if (*ret)
|
|
|
|
condition_free(*ret);
|
|
|
|
|
|
|
|
*ret = c;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-06 21:20:59 +02:00
|
|
|
int config_parse_ifnames(
|
|
|
|
const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
unsigned line,
|
|
|
|
const char *section,
|
|
|
|
unsigned section_line,
|
|
|
|
const char *lvalue,
|
|
|
|
int ltype,
|
|
|
|
const char *rvalue,
|
|
|
|
void *data,
|
|
|
|
void *userdata) {
|
2015-02-10 18:30:16 +01:00
|
|
|
|
|
|
|
char ***sv = data;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
assert(filename);
|
|
|
|
assert(lvalue);
|
|
|
|
assert(rvalue);
|
|
|
|
assert(data);
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
for (;;) {
|
|
|
|
_cleanup_free_ char *word = NULL;
|
2015-02-10 18:30:16 +01:00
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
r = extract_first_word(&rvalue, &word, NULL, 0);
|
2016-05-09 15:42:23 +02:00
|
|
|
if (r < 0) {
|
|
|
|
log_syntax(unit, LOG_ERR, filename, line, 0, "Failed to parse interface name list: %s", rvalue);
|
|
|
|
return 0;
|
|
|
|
}
|
2016-05-06 06:07:31 +02:00
|
|
|
if (r == 0)
|
|
|
|
break;
|
2015-02-10 18:30:16 +01:00
|
|
|
|
2016-05-06 21:20:59 +02:00
|
|
|
if (!ifname_valid(word)) {
|
|
|
|
log_syntax(unit, LOG_ERR, filename, line, 0, "Interface name is not valid or too long, ignoring assignment: %s", rvalue);
|
2015-02-10 18:30:16 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
r = strv_push(sv, word);
|
2015-02-10 18:30:16 +01:00
|
|
|
if (r < 0)
|
|
|
|
return log_oom();
|
2016-05-06 06:07:31 +02:00
|
|
|
|
|
|
|
word = NULL;
|
2015-02-10 18:30:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-11-25 01:33:04 +01:00
|
|
|
int config_parse_ifalias(const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
unsigned line,
|
|
|
|
const char *section,
|
2013-11-19 16:17:55 +01:00
|
|
|
unsigned section_line,
|
2013-11-25 01:33:04 +01:00
|
|
|
const char *lvalue,
|
|
|
|
int ltype,
|
|
|
|
const char *rvalue,
|
|
|
|
void *data,
|
|
|
|
void *userdata) {
|
|
|
|
|
|
|
|
char **s = data;
|
2015-01-10 00:33:46 +01:00
|
|
|
_cleanup_free_ char *n = NULL;
|
2013-11-25 01:33:04 +01:00
|
|
|
|
|
|
|
assert(filename);
|
|
|
|
assert(lvalue);
|
|
|
|
assert(rvalue);
|
|
|
|
assert(data);
|
|
|
|
|
|
|
|
n = strdup(rvalue);
|
|
|
|
if (!n)
|
|
|
|
return log_oom();
|
|
|
|
|
|
|
|
if (!ascii_is_valid(n) || strlen(n) >= IFALIASZ) {
|
2015-09-30 18:22:42 +02:00
|
|
|
log_syntax(unit, LOG_ERR, filename, line, 0, "Interface alias is not ASCII clean or is too long, ignoring assignment: %s", rvalue);
|
2013-11-25 01:33:04 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
free(*s);
|
2018-03-22 16:53:26 +01:00
|
|
|
if (*n)
|
|
|
|
*s = TAKE_PTR(n);
|
|
|
|
else
|
2013-11-25 01:33:04 +01:00
|
|
|
*s = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-10-28 20:59:56 +01:00
|
|
|
int config_parse_hwaddr(const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
unsigned line,
|
|
|
|
const char *section,
|
2013-11-19 16:17:55 +01:00
|
|
|
unsigned section_line,
|
2013-10-28 20:59:56 +01:00
|
|
|
const char *lvalue,
|
|
|
|
int ltype,
|
|
|
|
const char *rvalue,
|
|
|
|
void *data,
|
|
|
|
void *userdata) {
|
|
|
|
struct ether_addr **hwaddr = data;
|
|
|
|
struct ether_addr *n;
|
2016-05-17 22:43:36 +02:00
|
|
|
const char *start;
|
|
|
|
size_t offset;
|
2013-10-28 20:59:56 +01:00
|
|
|
int r;
|
|
|
|
|
|
|
|
assert(filename);
|
|
|
|
assert(lvalue);
|
|
|
|
assert(rvalue);
|
|
|
|
assert(data);
|
|
|
|
|
2013-10-29 15:59:45 +01:00
|
|
|
n = new0(struct ether_addr, 1);
|
2013-10-28 20:59:56 +01:00
|
|
|
if (!n)
|
|
|
|
return log_oom();
|
|
|
|
|
2016-05-17 22:43:36 +02:00
|
|
|
start = rvalue + strspn(rvalue, WHITESPACE);
|
|
|
|
r = ether_addr_from_string(start, n, &offset);
|
|
|
|
|
|
|
|
if (r || (start[offset + strspn(start + offset, WHITESPACE)] != '\0')) {
|
2015-09-30 18:22:42 +02:00
|
|
|
log_syntax(unit, LOG_ERR, filename, line, 0, "Not a valid MAC address, ignoring assignment: %s", rvalue);
|
2013-10-28 20:59:56 +01:00
|
|
|
free(n);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
free(*hwaddr);
|
|
|
|
*hwaddr = n;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2013-11-09 22:19:42 +01:00
|
|
|
|
2016-03-31 01:33:55 +02:00
|
|
|
int config_parse_iaid(const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
unsigned line,
|
|
|
|
const char *section,
|
|
|
|
unsigned section_line,
|
|
|
|
const char *lvalue,
|
|
|
|
int ltype,
|
|
|
|
const char *rvalue,
|
|
|
|
void *data,
|
|
|
|
void *userdata) {
|
|
|
|
uint32_t iaid;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
assert(filename);
|
|
|
|
assert(lvalue);
|
|
|
|
assert(rvalue);
|
|
|
|
assert(data);
|
|
|
|
|
|
|
|
r = safe_atou32(rvalue, &iaid);
|
|
|
|
if (r < 0) {
|
2016-04-05 00:13:06 +02:00
|
|
|
log_syntax(unit, LOG_ERR, filename, line, r,
|
|
|
|
"Unable to read IAID, ignoring assignment: %s", rvalue);
|
|
|
|
return 0;
|
2016-03-31 01:33:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
*((uint32_t *)data) = iaid;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-02 21:32:42 +02:00
|
|
|
int config_parse_bridge_port_priority(
|
|
|
|
const char *unit,
|
|
|
|
const char *filename,
|
|
|
|
unsigned line,
|
|
|
|
const char *section,
|
|
|
|
unsigned section_line,
|
|
|
|
const char *lvalue,
|
|
|
|
int ltype,
|
|
|
|
const char *rvalue,
|
|
|
|
void *data,
|
|
|
|
void *userdata) {
|
|
|
|
|
|
|
|
uint16_t i;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
assert(filename);
|
|
|
|
assert(lvalue);
|
|
|
|
assert(rvalue);
|
|
|
|
assert(data);
|
|
|
|
|
|
|
|
r = safe_atou16(rvalue, &i);
|
|
|
|
if (r < 0) {
|
|
|
|
log_syntax(unit, LOG_ERR, filename, line, r,
|
|
|
|
"Failed to parse bridge port priority, ignoring: %s", rvalue);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i > LINK_BRIDGE_PORT_PRIORITY_MAX) {
|
|
|
|
log_syntax(unit, LOG_ERR, filename, line, r,
|
|
|
|
"Bridge port priority is larger than maximum %u, ignoring: %s", LINK_BRIDGE_PORT_PRIORITY_MAX, rvalue);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
*((uint16_t *)data) = i;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-23 13:48:18 +02:00
|
|
|
void serialize_in_addrs(FILE *f, const struct in_addr *addresses, size_t size) {
|
2014-05-18 22:02:42 +02:00
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
assert(f);
|
|
|
|
assert(addresses);
|
|
|
|
assert(size);
|
|
|
|
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
fprintf(f, "%s%s", inet_ntoa(addresses[i]),
|
|
|
|
(i < (size - 1)) ? " ": "");
|
|
|
|
}
|
|
|
|
|
2014-07-17 01:39:46 +02:00
|
|
|
int deserialize_in_addrs(struct in_addr **ret, const char *string) {
|
2014-05-18 22:02:42 +02:00
|
|
|
_cleanup_free_ struct in_addr *addresses = NULL;
|
2014-07-17 01:39:46 +02:00
|
|
|
int size = 0;
|
2014-05-18 22:02:42 +02:00
|
|
|
|
|
|
|
assert(ret);
|
|
|
|
assert(string);
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
for (;;) {
|
|
|
|
_cleanup_free_ char *word = NULL;
|
2014-05-18 22:02:42 +02:00
|
|
|
struct in_addr *new_addresses;
|
|
|
|
int r;
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
r = extract_first_word(&string, &word, NULL, 0);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (r == 0)
|
|
|
|
break;
|
|
|
|
|
2018-02-27 19:09:22 +01:00
|
|
|
new_addresses = reallocarray(addresses, size + 1, sizeof(struct in_addr));
|
2014-05-18 22:02:42 +02:00
|
|
|
if (!new_addresses)
|
|
|
|
return -ENOMEM;
|
|
|
|
else
|
|
|
|
addresses = new_addresses;
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
r = inet_pton(AF_INET, word, &(addresses[size]));
|
2014-05-18 22:02:42 +02:00
|
|
|
if (r <= 0)
|
|
|
|
continue;
|
|
|
|
|
2016-02-23 05:32:04 +01:00
|
|
|
size++;
|
2014-05-18 22:02:42 +02:00
|
|
|
}
|
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*ret = TAKE_PTR(addresses);
|
2014-05-18 22:02:42 +02:00
|
|
|
|
2014-07-17 01:39:46 +02:00
|
|
|
return size;
|
2014-05-18 22:02:42 +02:00
|
|
|
}
|
|
|
|
|
2016-06-02 20:35:13 +02:00
|
|
|
void serialize_in6_addrs(FILE *f, const struct in6_addr *addresses, size_t size) {
|
2015-07-06 11:50:47 +02:00
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
assert(f);
|
|
|
|
assert(addresses);
|
|
|
|
assert(size);
|
|
|
|
|
2016-06-02 20:35:13 +02:00
|
|
|
for (i = 0; i < size; i++) {
|
|
|
|
char buffer[INET6_ADDRSTRLEN];
|
|
|
|
|
|
|
|
fputs(inet_ntop(AF_INET6, addresses+i, buffer, sizeof(buffer)), f);
|
|
|
|
|
|
|
|
if (i < size - 1)
|
|
|
|
fputc(' ', f);
|
|
|
|
}
|
2015-07-06 11:50:47 +02:00
|
|
|
}
|
|
|
|
|
2014-07-17 01:39:46 +02:00
|
|
|
int deserialize_in6_addrs(struct in6_addr **ret, const char *string) {
|
2014-05-18 22:02:42 +02:00
|
|
|
_cleanup_free_ struct in6_addr *addresses = NULL;
|
2014-07-17 01:39:46 +02:00
|
|
|
int size = 0;
|
2014-05-18 22:02:42 +02:00
|
|
|
|
|
|
|
assert(ret);
|
|
|
|
assert(string);
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
for (;;) {
|
|
|
|
_cleanup_free_ char *word = NULL;
|
2014-05-18 22:02:42 +02:00
|
|
|
struct in6_addr *new_addresses;
|
|
|
|
int r;
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
r = extract_first_word(&string, &word, NULL, 0);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (r == 0)
|
|
|
|
break;
|
|
|
|
|
2018-02-27 19:09:22 +01:00
|
|
|
new_addresses = reallocarray(addresses, size + 1, sizeof(struct in6_addr));
|
2014-05-18 22:02:42 +02:00
|
|
|
if (!new_addresses)
|
|
|
|
return -ENOMEM;
|
|
|
|
else
|
|
|
|
addresses = new_addresses;
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
r = inet_pton(AF_INET6, word, &(addresses[size]));
|
2014-05-18 22:02:42 +02:00
|
|
|
if (r <= 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
size++;
|
|
|
|
}
|
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*ret = TAKE_PTR(addresses);
|
2014-05-18 22:02:42 +02:00
|
|
|
|
2014-07-17 01:39:46 +02:00
|
|
|
return size;
|
2014-05-18 22:02:42 +02:00
|
|
|
}
|
2014-06-28 00:00:06 +02:00
|
|
|
|
2016-01-20 14:44:14 +01:00
|
|
|
void serialize_dhcp_routes(FILE *f, const char *key, sd_dhcp_route **routes, size_t size) {
|
2014-06-28 00:00:06 +02:00
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
assert(f);
|
|
|
|
assert(key);
|
|
|
|
assert(routes);
|
|
|
|
assert(size);
|
|
|
|
|
|
|
|
fprintf(f, "%s=", key);
|
|
|
|
|
2014-11-18 14:59:42 +01:00
|
|
|
for (i = 0; i < size; i++) {
|
2016-01-20 14:44:14 +01:00
|
|
|
struct in_addr dest, gw;
|
|
|
|
uint8_t length;
|
|
|
|
|
|
|
|
assert_se(sd_dhcp_route_get_destination(routes[i], &dest) >= 0);
|
|
|
|
assert_se(sd_dhcp_route_get_gateway(routes[i], &gw) >= 0);
|
|
|
|
assert_se(sd_dhcp_route_get_destination_prefix_length(routes[i], &length) >= 0);
|
|
|
|
|
|
|
|
fprintf(f, "%s/%" PRIu8, inet_ntoa(dest), length);
|
|
|
|
fprintf(f, ",%s%s", inet_ntoa(gw), (i < (size - 1)) ? " ": "");
|
2014-11-18 14:59:42 +01:00
|
|
|
}
|
2014-06-28 00:00:06 +02:00
|
|
|
|
|
|
|
fputs("\n", f);
|
|
|
|
}
|
|
|
|
|
|
|
|
int deserialize_dhcp_routes(struct sd_dhcp_route **ret, size_t *ret_size, size_t *ret_allocated, const char *string) {
|
|
|
|
_cleanup_free_ struct sd_dhcp_route *routes = NULL;
|
|
|
|
size_t size = 0, allocated = 0;
|
|
|
|
|
|
|
|
assert(ret);
|
|
|
|
assert(ret_size);
|
|
|
|
assert(ret_allocated);
|
|
|
|
assert(string);
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
/* WORD FORMAT: dst_ip/dst_prefixlen,gw_ip */
|
|
|
|
for (;;) {
|
|
|
|
_cleanup_free_ char *word = NULL;
|
2014-06-28 00:00:06 +02:00
|
|
|
char *tok, *tok_end;
|
|
|
|
unsigned n;
|
|
|
|
int r;
|
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
r = extract_first_word(&string, &word, NULL, 0);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (r == 0)
|
|
|
|
break;
|
2014-06-28 00:00:06 +02:00
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
if (!GREEDY_REALLOC(routes, allocated, size + 1))
|
2014-06-29 21:39:08 +02:00
|
|
|
return -ENOMEM;
|
2014-06-28 00:00:06 +02:00
|
|
|
|
2016-05-06 06:07:31 +02:00
|
|
|
tok = word;
|
2014-06-28 00:00:06 +02:00
|
|
|
|
|
|
|
/* get the subnet */
|
|
|
|
tok_end = strchr(tok, '/');
|
|
|
|
if (!tok_end)
|
|
|
|
continue;
|
|
|
|
*tok_end = '\0';
|
|
|
|
|
|
|
|
r = inet_aton(tok, &routes[size].dst_addr);
|
|
|
|
if (r == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
tok = tok_end + 1;
|
|
|
|
|
|
|
|
/* get the prefixlen */
|
|
|
|
tok_end = strchr(tok, ',');
|
|
|
|
if (!tok_end)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
*tok_end = '\0';
|
|
|
|
|
|
|
|
r = safe_atou(tok, &n);
|
|
|
|
if (r < 0 || n > 32)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
routes[size].dst_prefixlen = (uint8_t) n;
|
|
|
|
tok = tok_end + 1;
|
|
|
|
|
|
|
|
/* get the gateway */
|
|
|
|
r = inet_aton(tok, &routes[size].gw_addr);
|
|
|
|
if (r == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
size++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*ret_size = size;
|
|
|
|
*ret_allocated = allocated;
|
2018-03-22 16:53:26 +01:00
|
|
|
*ret = TAKE_PTR(routes);
|
2014-06-28 00:00:06 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2015-08-01 05:18:51 +02:00
|
|
|
|
2015-08-26 23:05:34 +02:00
|
|
|
int serialize_dhcp_option(FILE *f, const char *key, const void *data, size_t size) {
|
2015-08-01 05:18:51 +02:00
|
|
|
_cleanup_free_ char *hex_buf = NULL;
|
|
|
|
|
|
|
|
assert(f);
|
|
|
|
assert(key);
|
|
|
|
assert(data);
|
|
|
|
|
|
|
|
hex_buf = hexmem(data, size);
|
|
|
|
if (hex_buf == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
fprintf(f, "%s=%s\n", key, hex_buf);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-08-26 23:05:34 +02:00
|
|
|
int deserialize_dhcp_option(void **data, size_t *data_len, const char *string) {
|
2015-08-01 05:18:51 +02:00
|
|
|
assert(data);
|
|
|
|
assert(data_len);
|
|
|
|
assert(string);
|
|
|
|
|
|
|
|
if (strlen(string) % 2)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
return unhexmem(string, strlen(string), (void **)data, data_len);
|
|
|
|
}
|