2017-11-18 17:09:20 +01:00
|
|
|
/* SPDX-License-Identifier: LGPL-2.1+ */
|
2014-02-27 01:24:05 +01:00
|
|
|
/***
|
|
|
|
This file is part of systemd.
|
|
|
|
|
|
|
|
Copyright 2011 Lennart Poettering
|
|
|
|
Copyright 2014 Tom Gundersen
|
|
|
|
|
|
|
|
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/>.
|
|
|
|
***/
|
|
|
|
|
|
|
|
#include <errno.h>
|
2015-02-12 14:06:32 +01:00
|
|
|
#include <poll.h>
|
2015-10-24 22:58:24 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <sys/inotify.h>
|
2014-02-27 01:24:05 +01:00
|
|
|
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "sd-network.h"
|
|
|
|
|
2015-10-27 03:01:06 +01:00
|
|
|
#include "alloc-util.h"
|
2015-10-27 14:58:05 +01:00
|
|
|
#include "fd-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "fileio.h"
|
2015-10-27 14:58:05 +01:00
|
|
|
#include "fs-util.h"
|
2014-02-27 01:24:05 +01:00
|
|
|
#include "macro.h"
|
2015-10-27 14:58:05 +01:00
|
|
|
#include "parse-util.h"
|
2016-02-20 22:06:12 +01:00
|
|
|
#include "stdio-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "string-util.h"
|
2014-02-27 01:24:05 +01:00
|
|
|
#include "strv.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "util.h"
|
2014-02-27 01:24:05 +01:00
|
|
|
|
2014-08-12 15:05:21 +02:00
|
|
|
_public_ int sd_network_get_operational_state(char **state) {
|
|
|
|
_cleanup_free_ char *s = NULL;
|
2014-05-07 16:35:05 +02:00
|
|
|
int r;
|
|
|
|
|
2014-02-27 01:24:05 +01:00
|
|
|
assert_return(state, -EINVAL);
|
|
|
|
|
2014-08-12 15:05:21 +02:00
|
|
|
r = parse_env_file("/run/systemd/netif/state", NEWLINE, "OPER_STATE", &s, NULL);
|
2014-02-28 02:07:29 +01:00
|
|
|
if (r == -ENOENT)
|
|
|
|
return -ENODATA;
|
2014-08-12 15:05:21 +02:00
|
|
|
if (r < 0)
|
2014-02-27 01:24:05 +01:00
|
|
|
return r;
|
2014-08-12 15:05:21 +02:00
|
|
|
if (isempty(s))
|
|
|
|
return -ENODATA;
|
2014-05-07 16:35:05 +02:00
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*state = TAKE_PTR(s);
|
2014-05-07 16:35:05 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-08-12 15:05:21 +02:00
|
|
|
static int network_get_strv(const char *key, char ***ret) {
|
|
|
|
_cleanup_strv_free_ char **a = NULL;
|
2014-05-08 17:21:37 +02:00
|
|
|
_cleanup_free_ char *s = NULL;
|
|
|
|
int r;
|
|
|
|
|
2014-08-12 15:05:21 +02:00
|
|
|
assert_return(ret, -EINVAL);
|
|
|
|
|
|
|
|
r = parse_env_file("/run/systemd/netif/state", NEWLINE, key, &s, NULL);
|
|
|
|
if (r == -ENOENT)
|
|
|
|
return -ENODATA;
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (isempty(s)) {
|
|
|
|
*ret = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
a = strv_split(s, " ");
|
|
|
|
if (!a)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
strv_uniq(a);
|
|
|
|
r = strv_length(a);
|
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*ret = TAKE_PTR(a);
|
2014-08-12 15:05:21 +02:00
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
_public_ int sd_network_get_dns(char ***ret) {
|
|
|
|
return network_get_strv("DNS", ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
_public_ int sd_network_get_ntp(char ***ret) {
|
|
|
|
return network_get_strv("NTP", ret);
|
|
|
|
}
|
|
|
|
|
networkd: rework Domains= setting
Previously, .network files only knew a vaguely defined "Domains=" concept, for which the documentation declared it was
the "DNS domain" for the network connection, without specifying what that means.
With this the Domains setting is reworked, so that there are now "routing" domains and "search" domains. The former are
to be used by resolved to route DNS request to specific network interfaces, the latter is to be used for searching
single-label hostnames with (in addition to being used for routing). Both settings are configured in the "Domains="
setting. Normal domain names listed in it are now considered search domains (for compatibility with existing setups),
while those prefixed with "~" are considered routing domains only. To route all lookups to a specific interface the
routing domain "." may be used, referring to the root domain. An alternative syntax for this is the "*", as was already
implemented before using the "wildcard" domain concept.
This commit adds proper parsers for this new logic, and exposes this via the sd-network API. This information is not
used by resolved yet, this will be added in a later commit.
2016-01-25 19:46:00 +01:00
|
|
|
_public_ int sd_network_get_search_domains(char ***ret) {
|
2014-08-15 16:02:14 +02:00
|
|
|
return network_get_strv("DOMAINS", ret);
|
|
|
|
}
|
|
|
|
|
networkd: rework Domains= setting
Previously, .network files only knew a vaguely defined "Domains=" concept, for which the documentation declared it was
the "DNS domain" for the network connection, without specifying what that means.
With this the Domains setting is reworked, so that there are now "routing" domains and "search" domains. The former are
to be used by resolved to route DNS request to specific network interfaces, the latter is to be used for searching
single-label hostnames with (in addition to being used for routing). Both settings are configured in the "Domains="
setting. Normal domain names listed in it are now considered search domains (for compatibility with existing setups),
while those prefixed with "~" are considered routing domains only. To route all lookups to a specific interface the
routing domain "." may be used, referring to the root domain. An alternative syntax for this is the "*", as was already
implemented before using the "wildcard" domain concept.
This commit adds proper parsers for this new logic, and exposes this via the sd-network API. This information is not
used by resolved yet, this will be added in a later commit.
2016-01-25 19:46:00 +01:00
|
|
|
_public_ int sd_network_get_route_domains(char ***ret) {
|
|
|
|
return network_get_strv("ROUTE_DOMAINS", ret);
|
|
|
|
}
|
|
|
|
|
2016-01-05 16:43:07 +01:00
|
|
|
static int network_link_get_string(int ifindex, const char *field, char **ret) {
|
2017-12-14 19:02:29 +01:00
|
|
|
char path[STRLEN("/run/systemd/netif/links/") + DECIMAL_STR_MAX(ifindex) + 1];
|
2016-02-20 22:06:12 +01:00
|
|
|
_cleanup_free_ char *s = NULL;
|
2014-08-12 15:05:21 +02:00
|
|
|
int r;
|
|
|
|
|
|
|
|
assert_return(ifindex > 0, -EINVAL);
|
2016-01-05 16:43:07 +01:00
|
|
|
assert_return(ret, -EINVAL);
|
2014-05-08 17:21:37 +02:00
|
|
|
|
2016-02-20 22:06:12 +01:00
|
|
|
xsprintf(path, "/run/systemd/netif/links/%i", ifindex);
|
2014-08-12 15:05:21 +02:00
|
|
|
|
2016-02-20 22:06:12 +01:00
|
|
|
r = parse_env_file(path, NEWLINE, field, &s, NULL);
|
2014-05-08 17:21:37 +02:00
|
|
|
if (r == -ENOENT)
|
|
|
|
return -ENODATA;
|
2014-08-12 15:05:21 +02:00
|
|
|
if (r < 0)
|
2014-05-08 17:21:37 +02:00
|
|
|
return r;
|
2014-08-12 15:05:21 +02:00
|
|
|
if (isempty(s))
|
|
|
|
return -ENODATA;
|
2014-05-08 17:21:37 +02:00
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*ret = TAKE_PTR(s);
|
2014-05-08 17:21:37 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-01-06 18:36:32 +01:00
|
|
|
static int network_link_get_strv(int ifindex, const char *key, char ***ret) {
|
2017-12-14 19:02:29 +01:00
|
|
|
char path[STRLEN("/run/systemd/netif/links/") + DECIMAL_STR_MAX(ifindex) + 1];
|
2016-01-06 18:36:32 +01:00
|
|
|
_cleanup_strv_free_ char **a = NULL;
|
2016-02-20 22:06:12 +01:00
|
|
|
_cleanup_free_ char *s = NULL;
|
2016-01-06 18:36:32 +01:00
|
|
|
int r;
|
|
|
|
|
|
|
|
assert_return(ifindex > 0, -EINVAL);
|
|
|
|
assert_return(ret, -EINVAL);
|
|
|
|
|
2016-02-20 22:06:12 +01:00
|
|
|
xsprintf(path, "/run/systemd/netif/links/%i", ifindex);
|
|
|
|
r = parse_env_file(path, NEWLINE, key, &s, NULL);
|
2016-01-06 18:36:32 +01:00
|
|
|
if (r == -ENOENT)
|
|
|
|
return -ENODATA;
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (isempty(s)) {
|
|
|
|
*ret = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
a = strv_split(s, " ");
|
|
|
|
if (!a)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
strv_uniq(a);
|
|
|
|
r = strv_length(a);
|
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*ret = TAKE_PTR(a);
|
2016-01-06 18:36:32 +01:00
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2016-01-05 16:43:07 +01:00
|
|
|
_public_ int sd_network_link_get_setup_state(int ifindex, char **state) {
|
|
|
|
return network_link_get_string(ifindex, "ADMIN_STATE", state);
|
|
|
|
}
|
2014-09-08 13:50:52 +02:00
|
|
|
|
2016-01-05 16:43:07 +01:00
|
|
|
_public_ int sd_network_link_get_network_file(int ifindex, char **filename) {
|
|
|
|
return network_link_get_string(ifindex, "NETWORK_FILE", filename);
|
2014-09-08 13:50:52 +02:00
|
|
|
}
|
|
|
|
|
2014-08-13 22:37:45 +02:00
|
|
|
_public_ int sd_network_link_get_operational_state(int ifindex, char **state) {
|
2016-01-05 16:43:07 +01:00
|
|
|
return network_link_get_string(ifindex, "OPER_STATE", state);
|
2014-02-27 01:24:05 +01:00
|
|
|
}
|
|
|
|
|
2017-11-30 18:03:50 +01:00
|
|
|
_public_ int sd_network_link_get_required_for_online(int ifindex) {
|
|
|
|
_cleanup_free_ char *s = NULL;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
r = network_link_get_string(ifindex, "REQUIRED_FOR_ONLINE", &s);
|
|
|
|
if (r < 0) {
|
|
|
|
/* Handle -ENODATA as RequiredForOnline=yes, for compatibility */
|
|
|
|
if (r == -ENODATA)
|
|
|
|
return true;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
return parse_boolean(s);
|
|
|
|
}
|
|
|
|
|
2014-08-13 22:37:45 +02:00
|
|
|
_public_ int sd_network_link_get_llmnr(int ifindex, char **llmnr) {
|
2016-01-05 16:43:07 +01:00
|
|
|
return network_link_get_string(ifindex, "LLMNR", llmnr);
|
2014-08-03 18:45:07 +02:00
|
|
|
}
|
|
|
|
|
2016-01-05 17:32:25 +01:00
|
|
|
_public_ int sd_network_link_get_mdns(int ifindex, char **mdns) {
|
|
|
|
return network_link_get_string(ifindex, "MDNS", mdns);
|
|
|
|
}
|
|
|
|
|
2016-01-05 19:57:33 +01:00
|
|
|
_public_ int sd_network_link_get_dnssec(int ifindex, char **dnssec) {
|
|
|
|
return network_link_get_string(ifindex, "DNSSEC", dnssec);
|
|
|
|
}
|
|
|
|
|
2016-01-06 18:36:32 +01:00
|
|
|
_public_ int sd_network_link_get_dnssec_negative_trust_anchors(int ifindex, char ***nta) {
|
|
|
|
return network_link_get_strv(ifindex, "DNSSEC_NTA", nta);
|
|
|
|
}
|
|
|
|
|
sd-lldp: rework sd-lldp API
This reworks the sd-lldp substantially, simplifying things on one hand, and
extending the logic a bit on the other.
Specifically:
- Besides the sd_lldp object only one other object is maintained now,
sd_lldp_neighbor. It's used both as storage for literal LLDP packets, and for
maintainging info about peers in the database. Separation between packet, TLV
and chassis data is not maintained anymore. This should be a major
simplification.
- The sd-lldp API has been extended so that a couple of per-neighbor fields may
be queried directly, without iterating through the object. Other fields that
may appear multiple times, OTOH have to be iterated through.
- The maximum number of entries in the neighbor database is now configurable
during runtime.
- The generation of callbacks from sd_lldp objects is more restricted:
callbacks are only invoked when actual data changed.
- The TTL information is now hooked with a timer event, so that removals from
the neighbor database due to TTLs now result in a callback event.
- Querying LLDP neighbor database will now return a strictly ordered array, to
guarantee stability.
- A "capabilities" mask may now be configured, that selects what type of LLDP
neighbor data is collected. This may be used to restrict collection of LLDP
info about routers instead of all neighbors. This is now exposed via
networkd's LLDP= setting.
- sd-lldp's API to serialize the collected data to text files has been removed.
Instead, there's now an API to extract the raw binary data from LLDP neighbor
objects, as well as one to convert this raw binary data back to an LLDP
neighbor object. networkd will save this raw binary data to /run now, and the
client side can simply parse the information.
- support for parsing the more exotic TLVs has been removed, since we are not
using that. Instead there are now APIs to extract the raw data from TLVs.
Given how easy it is to parse the TLVs clients should do so now directly
instead of relying on our APIs for that.
- A lot of the APIs that parse out LLDP strings have been simplified so that
they actually return strings, instead of char arrays with a length. To deal
with possibly dangerous characters the strings are escaped if needed.
- APIs to extract and format the chassis and port IDs as strings has been
added.
- lldp.h has been simplified a lot. The enums are anonymous now, since they
were never used as enums, but simply as constants. Most definitions we don't
actually use ourselves have eben removed.
2016-02-19 17:58:52 +01:00
|
|
|
_public_ int sd_network_link_get_timezone(int ifindex, char **ret) {
|
2016-01-05 16:43:07 +01:00
|
|
|
return network_link_get_string(ifindex, "TIMEZONE", ret);
|
2015-08-26 19:19:32 +02:00
|
|
|
}
|
2014-12-11 05:29:55 +01:00
|
|
|
|
2014-08-13 22:37:45 +02:00
|
|
|
_public_ int sd_network_link_get_dns(int ifindex, char ***ret) {
|
2016-01-06 18:36:32 +01:00
|
|
|
return network_link_get_strv(ifindex, "DNS", ret);
|
2014-05-19 18:42:14 +02:00
|
|
|
}
|
|
|
|
|
2014-08-13 22:37:45 +02:00
|
|
|
_public_ int sd_network_link_get_ntp(int ifindex, char ***ret) {
|
2016-01-06 18:36:32 +01:00
|
|
|
return network_link_get_strv(ifindex, "NTP", ret);
|
2014-05-19 18:42:14 +02:00
|
|
|
}
|
|
|
|
|
networkd: rework Domains= setting
Previously, .network files only knew a vaguely defined "Domains=" concept, for which the documentation declared it was
the "DNS domain" for the network connection, without specifying what that means.
With this the Domains setting is reworked, so that there are now "routing" domains and "search" domains. The former are
to be used by resolved to route DNS request to specific network interfaces, the latter is to be used for searching
single-label hostnames with (in addition to being used for routing). Both settings are configured in the "Domains="
setting. Normal domain names listed in it are now considered search domains (for compatibility with existing setups),
while those prefixed with "~" are considered routing domains only. To route all lookups to a specific interface the
routing domain "." may be used, referring to the root domain. An alternative syntax for this is the "*", as was already
implemented before using the "wildcard" domain concept.
This commit adds proper parsers for this new logic, and exposes this via the sd-network API. This information is not
used by resolved yet, this will be added in a later commit.
2016-01-25 19:46:00 +01:00
|
|
|
_public_ int sd_network_link_get_search_domains(int ifindex, char ***ret) {
|
2016-01-06 18:36:32 +01:00
|
|
|
return network_link_get_strv(ifindex, "DOMAINS", ret);
|
2014-08-14 20:35:37 +02:00
|
|
|
}
|
|
|
|
|
networkd: rework Domains= setting
Previously, .network files only knew a vaguely defined "Domains=" concept, for which the documentation declared it was
the "DNS domain" for the network connection, without specifying what that means.
With this the Domains setting is reworked, so that there are now "routing" domains and "search" domains. The former are
to be used by resolved to route DNS request to specific network interfaces, the latter is to be used for searching
single-label hostnames with (in addition to being used for routing). Both settings are configured in the "Domains="
setting. Normal domain names listed in it are now considered search domains (for compatibility with existing setups),
while those prefixed with "~" are considered routing domains only. To route all lookups to a specific interface the
routing domain "." may be used, referring to the root domain. An alternative syntax for this is the "*", as was already
implemented before using the "wildcard" domain concept.
This commit adds proper parsers for this new logic, and exposes this via the sd-network API. This information is not
used by resolved yet, this will be added in a later commit.
2016-01-25 19:46:00 +01:00
|
|
|
_public_ int sd_network_link_get_route_domains(int ifindex, char ***ret) {
|
|
|
|
return network_link_get_strv(ifindex, "ROUTE_DOMAINS", ret);
|
|
|
|
}
|
|
|
|
|
2016-02-19 20:43:03 +01:00
|
|
|
static int network_link_get_ifindexes(int ifindex, const char *key, int **ret) {
|
2017-12-14 19:02:29 +01:00
|
|
|
char path[STRLEN("/run/systemd/netif/links/") + DECIMAL_STR_MAX(ifindex) + 1];
|
2016-02-19 20:43:03 +01:00
|
|
|
_cleanup_free_ int *ifis = NULL;
|
2016-02-20 22:06:12 +01:00
|
|
|
_cleanup_free_ char *s = NULL;
|
2016-02-19 20:43:03 +01:00
|
|
|
size_t allocated = 0, c = 0;
|
|
|
|
const char *x;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
assert_return(ifindex > 0, -EINVAL);
|
|
|
|
assert_return(ret, -EINVAL);
|
|
|
|
|
2016-02-20 22:06:12 +01:00
|
|
|
xsprintf(path, "/run/systemd/netif/links/%i", ifindex);
|
|
|
|
r = parse_env_file(path, NEWLINE, key, &s, NULL);
|
2016-02-19 20:43:03 +01:00
|
|
|
if (r == -ENOENT)
|
|
|
|
return -ENODATA;
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
|
2016-11-11 18:58:41 +01:00
|
|
|
for (x = s;;) {
|
2016-02-19 20:43:03 +01:00
|
|
|
_cleanup_free_ char *word = NULL;
|
|
|
|
|
|
|
|
r = extract_first_word(&x, &word, NULL, 0);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
if (r == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
r = parse_ifindex(word, &ifindex);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
|
2016-11-11 18:58:41 +01:00
|
|
|
if (!GREEDY_REALLOC(ifis, allocated, c + 2))
|
2016-02-19 20:43:03 +01:00
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
ifis[c++] = ifindex;
|
|
|
|
}
|
|
|
|
|
2016-11-11 18:58:41 +01:00
|
|
|
if (ifis)
|
2017-02-24 18:14:02 +01:00
|
|
|
ifis[c] = 0; /* Let's add a 0 ifindex to the end, to be nice */
|
2016-02-19 20:43:03 +01:00
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*ret = TAKE_PTR(ifis);
|
2016-02-19 20:43:03 +01:00
|
|
|
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
_public_ int sd_network_link_get_carrier_bound_to(int ifindex, int **ret) {
|
|
|
|
return network_link_get_ifindexes(ifindex, "CARRIER_BOUND_TO", ret);
|
2015-02-17 13:06:57 +01:00
|
|
|
}
|
|
|
|
|
2016-02-19 20:43:03 +01:00
|
|
|
_public_ int sd_network_link_get_carrier_bound_by(int ifindex, int **ret) {
|
|
|
|
return network_link_get_ifindexes(ifindex, "CARRIER_BOUND_BY", ret);
|
2015-02-17 13:06:57 +01:00
|
|
|
}
|
|
|
|
|
2014-02-27 01:24:05 +01:00
|
|
|
static inline int MONITOR_TO_FD(sd_network_monitor *m) {
|
|
|
|
return (int) (unsigned long) m - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline sd_network_monitor* FD_TO_MONITOR(int fd) {
|
|
|
|
return (sd_network_monitor*) (unsigned long) (fd + 1);
|
|
|
|
}
|
|
|
|
|
2015-06-10 19:00:39 +02:00
|
|
|
static int monitor_add_inotify_watch(int fd) {
|
|
|
|
int k;
|
|
|
|
|
|
|
|
k = inotify_add_watch(fd, "/run/systemd/netif/links/", IN_MOVED_TO|IN_DELETE);
|
|
|
|
if (k >= 0)
|
|
|
|
return 0;
|
|
|
|
else if (errno != ENOENT)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
k = inotify_add_watch(fd, "/run/systemd/netif/", IN_CREATE|IN_ISDIR);
|
|
|
|
if (k >= 0)
|
|
|
|
return 0;
|
|
|
|
else if (errno != ENOENT)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
k = inotify_add_watch(fd, "/run/systemd/", IN_CREATE|IN_ISDIR);
|
|
|
|
if (k < 0)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-17 01:07:17 +02:00
|
|
|
_public_ int sd_network_monitor_new(sd_network_monitor **m, const char *category) {
|
2015-06-10 19:00:39 +02:00
|
|
|
_cleanup_close_ int fd = -1;
|
|
|
|
int k;
|
2014-02-27 01:24:05 +01:00
|
|
|
bool good = false;
|
|
|
|
|
|
|
|
assert_return(m, -EINVAL);
|
|
|
|
|
|
|
|
fd = inotify_init1(IN_NONBLOCK|IN_CLOEXEC);
|
|
|
|
if (fd < 0)
|
|
|
|
return -errno;
|
|
|
|
|
2014-04-29 22:27:29 +02:00
|
|
|
if (!category || streq(category, "links")) {
|
2015-06-10 19:00:39 +02:00
|
|
|
k = monitor_add_inotify_watch(fd);
|
|
|
|
if (k < 0)
|
|
|
|
return k;
|
2014-02-27 01:24:05 +01:00
|
|
|
|
|
|
|
good = true;
|
|
|
|
}
|
|
|
|
|
2015-06-11 10:37:15 +02:00
|
|
|
if (!good)
|
2014-02-27 01:24:05 +01:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
*m = FD_TO_MONITOR(fd);
|
2015-06-10 19:00:39 +02:00
|
|
|
fd = -1;
|
|
|
|
|
2014-02-27 01:24:05 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
_public_ sd_network_monitor* sd_network_monitor_unref(sd_network_monitor *m) {
|
|
|
|
int fd;
|
|
|
|
|
2015-06-10 15:55:09 +02:00
|
|
|
if (m) {
|
|
|
|
fd = MONITOR_TO_FD(m);
|
|
|
|
close_nointr(fd);
|
|
|
|
}
|
2014-02-27 01:24:05 +01:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
_public_ int sd_network_monitor_flush(sd_network_monitor *m) {
|
2015-06-10 19:00:39 +02:00
|
|
|
union inotify_event_buffer buffer;
|
|
|
|
struct inotify_event *e;
|
|
|
|
ssize_t l;
|
|
|
|
int fd, k;
|
2014-02-27 01:24:05 +01:00
|
|
|
|
|
|
|
assert_return(m, -EINVAL);
|
|
|
|
|
2015-06-10 19:00:39 +02:00
|
|
|
fd = MONITOR_TO_FD(m);
|
|
|
|
|
|
|
|
l = read(fd, &buffer, sizeof(buffer));
|
|
|
|
if (l < 0) {
|
2017-09-28 10:17:04 +02:00
|
|
|
if (IN_SET(errno, EAGAIN, EINTR))
|
2015-06-10 19:00:39 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
return -errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
FOREACH_INOTIFY_EVENT(e, buffer, l) {
|
|
|
|
if (e->mask & IN_ISDIR) {
|
|
|
|
k = monitor_add_inotify_watch(fd);
|
|
|
|
if (k < 0)
|
|
|
|
return k;
|
|
|
|
|
|
|
|
k = inotify_rm_watch(fd, e->wd);
|
|
|
|
if (k < 0)
|
|
|
|
return -errno;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2014-02-27 01:24:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
_public_ int sd_network_monitor_get_fd(sd_network_monitor *m) {
|
|
|
|
|
|
|
|
assert_return(m, -EINVAL);
|
|
|
|
|
|
|
|
return MONITOR_TO_FD(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
_public_ int sd_network_monitor_get_events(sd_network_monitor *m) {
|
|
|
|
|
|
|
|
assert_return(m, -EINVAL);
|
|
|
|
|
|
|
|
/* For now we will only return POLLIN here, since we don't
|
|
|
|
* need anything else ever for inotify. However, let's have
|
|
|
|
* this API to keep our options open should we later on need
|
|
|
|
* it. */
|
|
|
|
return POLLIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
_public_ int sd_network_monitor_get_timeout(sd_network_monitor *m, uint64_t *timeout_usec) {
|
|
|
|
|
|
|
|
assert_return(m, -EINVAL);
|
|
|
|
assert_return(timeout_usec, -EINVAL);
|
|
|
|
|
|
|
|
/* For now we will only return (uint64_t) -1, since we don't
|
|
|
|
* need any timeout. However, let's have this API to keep our
|
|
|
|
* options open should we later on need it. */
|
|
|
|
*timeout_usec = (uint64_t) -1;
|
|
|
|
return 0;
|
|
|
|
}
|