mirror of
https://github.com/systemd/systemd.git
synced 2025-02-04 21:47:31 +03:00
Merge pull request #18717 from yuwata/sd-device-monitor-introduce-more-filters
sd-device-monitor: introduce two new filters and use them in dissect-image.c
This commit is contained in:
commit
bd6ea22920
@ -751,3 +751,9 @@ global:
|
||||
sd_device_new_from_stat_rdev;
|
||||
sd_device_trigger;
|
||||
} LIBSYSTEMD_247;
|
||||
|
||||
LIBSYSTEMD_249 {
|
||||
global:
|
||||
sd_device_monitor_filter_add_match_sysattr;
|
||||
sd_device_monitor_filter_add_match_parent;
|
||||
} LIBSYSTEMD_248;
|
||||
|
@ -127,6 +127,7 @@ libsystemd_sources = files('''
|
||||
sd-device/device-monitor.c
|
||||
sd-device/device-private.c
|
||||
sd-device/device-private.h
|
||||
sd-device/device-util.c
|
||||
sd-device/device-util.h
|
||||
sd-device/sd-device.c
|
||||
sd-hwdb/hwdb-internal.h
|
||||
|
@ -287,42 +287,6 @@ int device_enumerator_add_device(sd_device_enumerator *enumerator, sd_device *de
|
||||
return 0;
|
||||
}
|
||||
|
||||
static bool match_sysattr_value(sd_device *device, const char *sysattr, const char *match_value) {
|
||||
const char *value;
|
||||
|
||||
assert(device);
|
||||
assert(sysattr);
|
||||
|
||||
if (sd_device_get_sysattr_value(device, sysattr, &value) < 0)
|
||||
return false;
|
||||
|
||||
if (!match_value)
|
||||
return true;
|
||||
|
||||
if (fnmatch(match_value, value, 0) == 0)
|
||||
return true;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool match_sysattr(sd_device_enumerator *enumerator, sd_device *device) {
|
||||
const char *sysattr;
|
||||
const char *value;
|
||||
|
||||
assert(enumerator);
|
||||
assert(device);
|
||||
|
||||
HASHMAP_FOREACH_KEY(value, sysattr, enumerator->nomatch_sysattr)
|
||||
if (match_sysattr_value(device, sysattr, value))
|
||||
return false;
|
||||
|
||||
HASHMAP_FOREACH_KEY(value, sysattr, enumerator->match_sysattr)
|
||||
if (!match_sysattr_value(device, sysattr, value))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool match_property(sd_device_enumerator *enumerator, sd_device *device) {
|
||||
const char *property;
|
||||
const char *value;
|
||||
@ -367,24 +331,6 @@ static bool match_tag(sd_device_enumerator *enumerator, sd_device *device) {
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool match_parent(sd_device_enumerator *enumerator, sd_device *device) {
|
||||
const char *syspath_parent, *syspath;
|
||||
|
||||
assert(enumerator);
|
||||
assert(device);
|
||||
|
||||
if (set_isempty(enumerator->match_parent))
|
||||
return true;
|
||||
|
||||
assert_se(sd_device_get_syspath(device, &syspath) >= 0);
|
||||
|
||||
SET_FOREACH(syspath_parent, enumerator->match_parent)
|
||||
if (path_startswith(syspath, syspath_parent))
|
||||
return true;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool match_sysname(sd_device_enumerator *enumerator, const char *sysname) {
|
||||
const char *sysname_match;
|
||||
|
||||
@ -470,7 +416,7 @@ static int enumerator_scan_dir_and_add_devices(sd_device_enumerator *enumerator,
|
||||
sd_device_get_ifindex(device, NULL) >= 0))
|
||||
continue;
|
||||
|
||||
if (!match_parent(enumerator, device))
|
||||
if (!device_match_parent(device, enumerator->match_parent, NULL))
|
||||
continue;
|
||||
|
||||
if (!match_tag(enumerator, device))
|
||||
@ -479,7 +425,7 @@ static int enumerator_scan_dir_and_add_devices(sd_device_enumerator *enumerator,
|
||||
if (!match_property(enumerator, device))
|
||||
continue;
|
||||
|
||||
if (!match_sysattr(enumerator, device))
|
||||
if (!device_match_sysattr(device, enumerator->match_sysattr, enumerator->nomatch_sysattr))
|
||||
continue;
|
||||
|
||||
k = device_enumerator_add_device(enumerator, device);
|
||||
@ -600,13 +546,13 @@ static int enumerator_scan_devices_tag(sd_device_enumerator *enumerator, const c
|
||||
if (!match_sysname(enumerator, sysname))
|
||||
continue;
|
||||
|
||||
if (!match_parent(enumerator, device))
|
||||
if (!device_match_parent(device, enumerator->match_parent, NULL))
|
||||
continue;
|
||||
|
||||
if (!match_property(enumerator, device))
|
||||
continue;
|
||||
|
||||
if (!match_sysattr(enumerator, device))
|
||||
if (!device_match_sysattr(device, enumerator->match_sysattr, enumerator->nomatch_sysattr))
|
||||
continue;
|
||||
|
||||
k = device_enumerator_add_device(enumerator, device);
|
||||
@ -667,7 +613,7 @@ static int parent_add_child(sd_device_enumerator *enumerator, const char *path)
|
||||
if (!match_property(enumerator, device))
|
||||
return 0;
|
||||
|
||||
if (!match_sysattr(enumerator, device))
|
||||
if (!device_match_sysattr(device, enumerator->match_sysattr, enumerator->nomatch_sysattr))
|
||||
return 0;
|
||||
|
||||
r = device_enumerator_add_device(enumerator, device);
|
||||
|
@ -37,6 +37,10 @@ struct sd_device_monitor {
|
||||
|
||||
Hashmap *subsystem_filter;
|
||||
Set *tag_filter;
|
||||
Hashmap *match_sysattr_filter;
|
||||
Hashmap *nomatch_sysattr_filter;
|
||||
Set *match_parent_filter;
|
||||
Set *nomatch_parent_filter;
|
||||
bool filter_uptodate;
|
||||
|
||||
sd_event *event;
|
||||
@ -336,55 +340,78 @@ static sd_device_monitor *device_monitor_free(sd_device_monitor *m) {
|
||||
|
||||
hashmap_free(m->subsystem_filter);
|
||||
set_free(m->tag_filter);
|
||||
hashmap_free(m->match_sysattr_filter);
|
||||
hashmap_free(m->nomatch_sysattr_filter);
|
||||
set_free(m->match_parent_filter);
|
||||
set_free(m->nomatch_parent_filter);
|
||||
|
||||
return mfree(m);
|
||||
}
|
||||
|
||||
DEFINE_PUBLIC_TRIVIAL_REF_UNREF_FUNC(sd_device_monitor, sd_device_monitor, device_monitor_free);
|
||||
|
||||
static int passes_filter(sd_device_monitor *m, sd_device *device) {
|
||||
const char *tag, *subsystem, *devtype, *s, *d = NULL;
|
||||
static int check_subsystem_filter(sd_device_monitor *m, sd_device *device) {
|
||||
const char *s, *subsystem, *d, *devtype = NULL;
|
||||
int r;
|
||||
|
||||
assert(m);
|
||||
assert(device);
|
||||
|
||||
if (hashmap_isempty(m->subsystem_filter))
|
||||
goto tag;
|
||||
return true;
|
||||
|
||||
r = sd_device_get_subsystem(device, &s);
|
||||
r = sd_device_get_subsystem(device, &subsystem);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_get_devtype(device, &d);
|
||||
r = sd_device_get_devtype(device, &devtype);
|
||||
if (r < 0 && r != -ENOENT)
|
||||
return r;
|
||||
|
||||
HASHMAP_FOREACH_KEY(devtype, subsystem, m->subsystem_filter) {
|
||||
HASHMAP_FOREACH_KEY(d, s, m->subsystem_filter) {
|
||||
if (!streq(s, subsystem))
|
||||
continue;
|
||||
|
||||
if (!devtype)
|
||||
goto tag;
|
||||
|
||||
if (!d)
|
||||
continue;
|
||||
|
||||
if (streq(d, devtype))
|
||||
goto tag;
|
||||
if (!d || streq_ptr(d, devtype))
|
||||
return true;
|
||||
}
|
||||
|
||||
return 0;
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool check_tag_filter(sd_device_monitor *m, sd_device *device) {
|
||||
const char *tag;
|
||||
|
||||
assert(m);
|
||||
assert(device);
|
||||
|
||||
tag:
|
||||
if (set_isempty(m->tag_filter))
|
||||
return 1;
|
||||
return true;
|
||||
|
||||
SET_FOREACH(tag, m->tag_filter)
|
||||
if (sd_device_has_tag(device, tag) > 0)
|
||||
return 1;
|
||||
return true;
|
||||
|
||||
return 0;
|
||||
return false;
|
||||
}
|
||||
|
||||
static int passes_filter(sd_device_monitor *m, sd_device *device) {
|
||||
int r;
|
||||
|
||||
assert(m);
|
||||
assert(device);
|
||||
|
||||
r = check_subsystem_filter(m, device);
|
||||
if (r <= 0)
|
||||
return r;
|
||||
|
||||
if (!check_tag_filter(m, device))
|
||||
return false;
|
||||
|
||||
if (!device_match_sysattr(device, m->match_sysattr_filter, m->nomatch_sysattr_filter))
|
||||
return false;
|
||||
|
||||
return device_match_parent(device, m->match_parent_filter, m->nomatch_parent_filter);
|
||||
}
|
||||
|
||||
int device_monitor_receive_device(sd_device_monitor *m, sd_device **ret) {
|
||||
@ -747,6 +774,42 @@ _public_ int sd_device_monitor_filter_add_match_tag(sd_device_monitor *m, const
|
||||
return r;
|
||||
}
|
||||
|
||||
_public_ int sd_device_monitor_filter_add_match_sysattr(sd_device_monitor *m, const char *sysattr, const char *value, int match) {
|
||||
Hashmap **hashmap;
|
||||
|
||||
assert_return(m, -EINVAL);
|
||||
assert_return(sysattr, -EINVAL);
|
||||
|
||||
if (match)
|
||||
hashmap = &m->match_sysattr_filter;
|
||||
else
|
||||
hashmap = &m->nomatch_sysattr_filter;
|
||||
|
||||
/* TODO: unset m->filter_uptodate on success when we support this filter on BPF. */
|
||||
return hashmap_put_strdup_full(hashmap, &trivial_hash_ops_free_free, sysattr, value);
|
||||
}
|
||||
|
||||
_public_ int sd_device_monitor_filter_add_match_parent(sd_device_monitor *m, sd_device *device, int match) {
|
||||
const char *syspath;
|
||||
Set **set;
|
||||
int r;
|
||||
|
||||
assert_return(m, -EINVAL);
|
||||
assert_return(device, -EINVAL);
|
||||
|
||||
r = sd_device_get_syspath(device, &syspath);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (match)
|
||||
set = &m->match_parent_filter;
|
||||
else
|
||||
set = &m->nomatch_parent_filter;
|
||||
|
||||
/* TODO: unset m->filter_uptodate on success when we support this filter on BPF. */
|
||||
return set_put_strdup(set, syspath);
|
||||
}
|
||||
|
||||
_public_ int sd_device_monitor_filter_remove(sd_device_monitor *m) {
|
||||
static const struct sock_fprog filter = { 0, NULL };
|
||||
|
||||
@ -754,6 +817,10 @@ _public_ int sd_device_monitor_filter_remove(sd_device_monitor *m) {
|
||||
|
||||
m->subsystem_filter = hashmap_free(m->subsystem_filter);
|
||||
m->tag_filter = set_free(m->tag_filter);
|
||||
m->match_sysattr_filter = hashmap_free(m->match_sysattr_filter);
|
||||
m->nomatch_sysattr_filter = hashmap_free(m->nomatch_sysattr_filter);
|
||||
m->match_parent_filter = set_free(m->match_parent_filter);
|
||||
m->nomatch_parent_filter = set_free(m->nomatch_parent_filter);
|
||||
|
||||
if (setsockopt(m->sock, SOL_SOCKET, SO_DETACH_FILTER, &filter, sizeof(filter)) < 0)
|
||||
return -errno;
|
||||
|
63
src/libsystemd/sd-device/device-util.c
Normal file
63
src/libsystemd/sd-device/device-util.c
Normal file
@ -0,0 +1,63 @@
|
||||
/* SPDX-License-Identifier: LGPL-2.1-or-later */
|
||||
|
||||
#include <fnmatch.h>
|
||||
|
||||
#include "device-util.h"
|
||||
#include "path-util.h"
|
||||
|
||||
static bool device_match_sysattr_value(sd_device *device, const char *sysattr, const char *match_value) {
|
||||
const char *value;
|
||||
|
||||
assert(device);
|
||||
assert(sysattr);
|
||||
|
||||
if (sd_device_get_sysattr_value(device, sysattr, &value) < 0)
|
||||
return false;
|
||||
|
||||
if (!match_value)
|
||||
return true;
|
||||
|
||||
if (fnmatch(match_value, value, 0) == 0)
|
||||
return true;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool device_match_sysattr(sd_device *device, Hashmap *match_sysattr, Hashmap *nomatch_sysattr) {
|
||||
const char *sysattr;
|
||||
const char *value;
|
||||
|
||||
assert(device);
|
||||
|
||||
HASHMAP_FOREACH_KEY(value, sysattr, match_sysattr)
|
||||
if (!device_match_sysattr_value(device, sysattr, value))
|
||||
return false;
|
||||
|
||||
HASHMAP_FOREACH_KEY(value, sysattr, nomatch_sysattr)
|
||||
if (device_match_sysattr_value(device, sysattr, value))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool device_match_parent(sd_device *device, Set *match_parent, Set *nomatch_parent) {
|
||||
const char *syspath_parent, *syspath;
|
||||
|
||||
assert(device);
|
||||
|
||||
if (sd_device_get_syspath(device, &syspath) < 0)
|
||||
return false;
|
||||
|
||||
SET_FOREACH(syspath_parent, nomatch_parent)
|
||||
if (path_startswith(syspath, syspath_parent))
|
||||
return false;
|
||||
|
||||
if (set_isempty(match_parent))
|
||||
return true;
|
||||
|
||||
SET_FOREACH(syspath_parent, match_parent)
|
||||
if (path_startswith(syspath, syspath_parent))
|
||||
return true;
|
||||
|
||||
return false;
|
||||
}
|
@ -1,8 +1,15 @@
|
||||
/* SPDX-License-Identifier: LGPL-2.1-or-later */
|
||||
#pragma once
|
||||
|
||||
#include <stdbool.h>
|
||||
|
||||
#include "sd-device.h"
|
||||
|
||||
#include "hashmap.h"
|
||||
#include "log.h"
|
||||
#include "macro.h"
|
||||
#include "set.h"
|
||||
|
||||
#define FOREACH_DEVICE_PROPERTY(device, key, value) \
|
||||
for (key = sd_device_get_property_first(device, &(value)); \
|
||||
key; \
|
||||
@ -64,3 +71,6 @@
|
||||
#define log_device_notice_errno(device, error, ...) log_device_full_errno(device, LOG_NOTICE, error, __VA_ARGS__)
|
||||
#define log_device_warning_errno(device, error, ...) log_device_full_errno(device, LOG_WARNING, error, __VA_ARGS__)
|
||||
#define log_device_error_errno(device, error, ...) log_device_full_errno(device, LOG_ERR, error, __VA_ARGS__)
|
||||
|
||||
bool device_match_sysattr(sd_device *device, Hashmap *match_sysattr, Hashmap *nomatch_sysattr);
|
||||
bool device_match_parent(sd_device *device, Set *match_parent, Set *nomatch_parent);
|
||||
|
@ -95,10 +95,10 @@ static void test_send_receive_one(sd_device *device, bool subsystem_filter, bool
|
||||
static void test_subsystem_filter(sd_device *device) {
|
||||
_cleanup_(sd_device_monitor_unrefp) sd_device_monitor *monitor_server = NULL, *monitor_client = NULL;
|
||||
_cleanup_(sd_device_enumerator_unrefp) sd_device_enumerator *e = NULL;
|
||||
const char *syspath, *subsystem, *p, *s;
|
||||
const char *syspath, *subsystem;
|
||||
sd_device *d;
|
||||
|
||||
log_info("/* %s */", __func__);
|
||||
log_device_info(device, "/* %s */", __func__);
|
||||
|
||||
assert_se(sd_device_get_syspath(device, &syspath) >= 0);
|
||||
assert_se(sd_device_get_subsystem(device, &subsystem) >= 0);
|
||||
@ -116,18 +116,142 @@ static void test_subsystem_filter(sd_device *device) {
|
||||
assert_se(sd_device_enumerator_new(&e) >= 0);
|
||||
assert_se(sd_device_enumerator_add_match_subsystem(e, subsystem, false) >= 0);
|
||||
FOREACH_DEVICE(e, d) {
|
||||
const char *p, *s;
|
||||
|
||||
assert_se(sd_device_get_syspath(d, &p) >= 0);
|
||||
assert_se(sd_device_get_subsystem(d, &s) >= 0);
|
||||
|
||||
log_info("Sending device subsystem:%s syspath:%s", s, p);
|
||||
log_device_debug(d, "Sending device subsystem:%s syspath:%s", s, p);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, d) >= 0);
|
||||
}
|
||||
|
||||
log_info("Sending device subsystem:%s syspath:%s", subsystem, syspath);
|
||||
log_device_info(device, "Sending device subsystem:%s syspath:%s", subsystem, syspath);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, device) >= 0);
|
||||
assert_se(sd_event_loop(sd_device_monitor_get_event(monitor_client)) == 100);
|
||||
}
|
||||
|
||||
static void test_tag_filter(sd_device *device) {
|
||||
_cleanup_(sd_device_monitor_unrefp) sd_device_monitor *monitor_server = NULL, *monitor_client = NULL;
|
||||
_cleanup_(sd_device_enumerator_unrefp) sd_device_enumerator *e = NULL;
|
||||
const char *syspath;
|
||||
sd_device *d;
|
||||
|
||||
log_device_info(device, "/* %s */", __func__);
|
||||
|
||||
assert_se(sd_device_get_syspath(device, &syspath) >= 0);
|
||||
|
||||
assert_se(device_monitor_new_full(&monitor_server, MONITOR_GROUP_NONE, -1) >= 0);
|
||||
assert_se(sd_device_monitor_start(monitor_server, NULL, NULL) >= 0);
|
||||
assert_se(sd_event_source_set_description(sd_device_monitor_get_event_source(monitor_server), "sender") >= 0);
|
||||
|
||||
assert_se(device_monitor_new_full(&monitor_client, MONITOR_GROUP_NONE, -1) >= 0);
|
||||
assert_se(device_monitor_allow_unicast_sender(monitor_client, monitor_server) >= 0);
|
||||
assert_se(sd_device_monitor_filter_add_match_tag(monitor_client, "TEST_SD_DEVICE_MONITOR") >= 0);
|
||||
assert_se(sd_device_monitor_start(monitor_client, monitor_handler, (void *) syspath) >= 0);
|
||||
assert_se(sd_event_source_set_description(sd_device_monitor_get_event_source(monitor_client), "receiver") >= 0);
|
||||
|
||||
assert_se(sd_device_enumerator_new(&e) >= 0);
|
||||
FOREACH_DEVICE(e, d) {
|
||||
const char *p;
|
||||
|
||||
assert_se(sd_device_get_syspath(d, &p) >= 0);
|
||||
|
||||
log_device_debug(d, "Sending device syspath:%s", p);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, d) >= 0);
|
||||
}
|
||||
|
||||
log_device_info(device, "Sending device syspath:%s", syspath);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, device) >= 0);
|
||||
assert_se(sd_event_loop(sd_device_monitor_get_event(monitor_client)) == 100);
|
||||
|
||||
}
|
||||
|
||||
static void test_sysattr_filter(sd_device *device, const char *sysattr) {
|
||||
_cleanup_(sd_device_monitor_unrefp) sd_device_monitor *monitor_server = NULL, *monitor_client = NULL;
|
||||
_cleanup_(sd_device_enumerator_unrefp) sd_device_enumerator *e = NULL;
|
||||
const char *syspath, *subsystem, *sysattr_value;
|
||||
sd_device *d;
|
||||
|
||||
log_device_info(device, "/* %s(%s) */", __func__, sysattr);
|
||||
|
||||
assert_se(sd_device_get_syspath(device, &syspath) >= 0);
|
||||
assert_se(sd_device_get_subsystem(device, &subsystem) >= 0);
|
||||
assert_se(sd_device_get_sysattr_value(device, sysattr, &sysattr_value) >= 0);
|
||||
|
||||
assert_se(device_monitor_new_full(&monitor_server, MONITOR_GROUP_NONE, -1) >= 0);
|
||||
assert_se(sd_device_monitor_start(monitor_server, NULL, NULL) >= 0);
|
||||
assert_se(sd_event_source_set_description(sd_device_monitor_get_event_source(monitor_server), "sender") >= 0);
|
||||
|
||||
assert_se(device_monitor_new_full(&monitor_client, MONITOR_GROUP_NONE, -1) >= 0);
|
||||
assert_se(device_monitor_allow_unicast_sender(monitor_client, monitor_server) >= 0);
|
||||
/* The sysattr filter is not implemented in BPF yet, so the below device_monito_send_device()
|
||||
* may cause EAGAIN. So, let's also filter devices with subsystem. */
|
||||
assert_se(sd_device_monitor_filter_add_match_subsystem_devtype(monitor_client, subsystem, NULL) >= 0);
|
||||
assert_se(sd_device_monitor_filter_add_match_sysattr(monitor_client, sysattr, sysattr_value, true) >= 0);
|
||||
assert_se(sd_device_monitor_start(monitor_client, monitor_handler, (void *) syspath) >= 0);
|
||||
assert_se(sd_event_source_set_description(sd_device_monitor_get_event_source(monitor_client), "receiver") >= 0);
|
||||
|
||||
assert_se(sd_device_enumerator_new(&e) >= 0);
|
||||
assert_se(sd_device_enumerator_add_match_sysattr(e, sysattr, sysattr_value, false) >= 0);
|
||||
FOREACH_DEVICE(e, d) {
|
||||
const char *p;
|
||||
|
||||
assert_se(sd_device_get_syspath(d, &p) >= 0);
|
||||
|
||||
log_device_debug(d, "Sending device syspath:%s", p);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, d) >= 0);
|
||||
}
|
||||
|
||||
log_device_info(device, "Sending device syspath:%s", syspath);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, device) >= 0);
|
||||
assert_se(sd_event_loop(sd_device_monitor_get_event(monitor_client)) == 100);
|
||||
|
||||
}
|
||||
|
||||
static void test_parent_filter(sd_device *device) {
|
||||
_cleanup_(sd_device_monitor_unrefp) sd_device_monitor *monitor_server = NULL, *monitor_client = NULL;
|
||||
_cleanup_(sd_device_enumerator_unrefp) sd_device_enumerator *e = NULL;
|
||||
const char *syspath, *subsystem;
|
||||
sd_device *parent, *d;
|
||||
int r;
|
||||
|
||||
log_device_info(device, "/* %s */", __func__);
|
||||
|
||||
assert_se(sd_device_get_syspath(device, &syspath) >= 0);
|
||||
assert_se(sd_device_get_subsystem(device, &subsystem) >= 0);
|
||||
r = sd_device_get_parent(device, &parent);
|
||||
if (r < 0)
|
||||
return (void) log_device_info(device, "Device does not have parent, skipping.");
|
||||
|
||||
assert_se(device_monitor_new_full(&monitor_server, MONITOR_GROUP_NONE, -1) >= 0);
|
||||
assert_se(sd_device_monitor_start(monitor_server, NULL, NULL) >= 0);
|
||||
assert_se(sd_event_source_set_description(sd_device_monitor_get_event_source(monitor_server), "sender") >= 0);
|
||||
|
||||
assert_se(device_monitor_new_full(&monitor_client, MONITOR_GROUP_NONE, -1) >= 0);
|
||||
assert_se(device_monitor_allow_unicast_sender(monitor_client, monitor_server) >= 0);
|
||||
/* The parent filter is not implemented in BPF yet, so the below device_monito_send_device()
|
||||
* may cause EAGAIN. So, let's also filter devices with subsystem. */
|
||||
assert_se(sd_device_monitor_filter_add_match_subsystem_devtype(monitor_client, subsystem, NULL) >= 0);
|
||||
assert_se(sd_device_monitor_filter_add_match_parent(monitor_client, parent, true) >= 0);
|
||||
assert_se(sd_device_monitor_start(monitor_client, monitor_handler, (void *) syspath) >= 0);
|
||||
assert_se(sd_event_source_set_description(sd_device_monitor_get_event_source(monitor_client), "receiver") >= 0);
|
||||
|
||||
assert_se(sd_device_enumerator_new(&e) >= 0);
|
||||
FOREACH_DEVICE(e, d) {
|
||||
const char *p;
|
||||
|
||||
assert_se(sd_device_get_syspath(d, &p) >= 0);
|
||||
|
||||
log_device_debug(d, "Sending device syspath:%s", p);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, d) >= 0);
|
||||
}
|
||||
|
||||
log_device_info(device, "Sending device syspath:%s", syspath);
|
||||
assert_se(device_monitor_send_device(monitor_server, monitor_client, device) >= 0);
|
||||
assert_se(sd_event_loop(sd_device_monitor_get_event(monitor_client)) == 100);
|
||||
|
||||
}
|
||||
|
||||
static void test_sd_device_monitor_filter_remove(sd_device *device) {
|
||||
_cleanup_(sd_device_monitor_unrefp) sd_device_monitor *monitor_server = NULL, *monitor_client = NULL;
|
||||
const char *syspath;
|
||||
@ -184,6 +308,7 @@ int main(int argc, char *argv[]) {
|
||||
assert_se(sd_device_new_from_syspath(&loopback, "/sys/class/net/lo") >= 0);
|
||||
assert_se(device_add_property(loopback, "ACTION", "add") >= 0);
|
||||
assert_se(device_add_property(loopback, "SEQNUM", "10") >= 0);
|
||||
assert_se(device_add_tag(loopback, "TEST_SD_DEVICE_MONITOR", true) >= 0);
|
||||
|
||||
test_send_receive_one(loopback, false, false, false);
|
||||
test_send_receive_one(loopback, true, false, false);
|
||||
@ -194,6 +319,8 @@ int main(int argc, char *argv[]) {
|
||||
test_send_receive_one(loopback, true, true, true);
|
||||
|
||||
test_subsystem_filter(loopback);
|
||||
test_tag_filter(loopback);
|
||||
test_sysattr_filter(loopback, "ifindex");
|
||||
test_sd_device_monitor_filter_remove(loopback);
|
||||
test_device_copy_properties(loopback);
|
||||
|
||||
@ -214,5 +341,7 @@ int main(int argc, char *argv[]) {
|
||||
test_send_receive_one(sda, false, true, true);
|
||||
test_send_receive_one(sda, true, true, true);
|
||||
|
||||
test_parent_filter(sda);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -127,33 +127,58 @@ static int enumerator_for_parent(sd_device *d, sd_device_enumerator **ret) {
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_enumerator_add_match_subsystem(e, "block", true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_enumerator_add_match_parent(e, d);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_enumerator_add_match_sysattr(e, "partition", NULL, true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
*ret = TAKE_PTR(e);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int device_is_partition(sd_device *d, blkid_partition pp) {
|
||||
static int device_is_partition(sd_device *d, sd_device *expected_parent, blkid_partition pp) {
|
||||
const char *v, *parent_syspath, *expected_parent_syspath;
|
||||
blkid_loff_t bsize, bstart;
|
||||
uint64_t size, start;
|
||||
int partno, bpartno, r;
|
||||
const char *ss, *v;
|
||||
sd_device *parent;
|
||||
|
||||
assert(d);
|
||||
assert(expected_parent);
|
||||
assert(pp);
|
||||
|
||||
r = sd_device_get_subsystem(d, &ss);
|
||||
r = sd_device_get_subsystem(d, &v);
|
||||
if (r < 0)
|
||||
return r;
|
||||
if (!streq(ss, "block"))
|
||||
if (!streq(v, "block"))
|
||||
return false;
|
||||
|
||||
r = sd_device_get_sysattr_value(d, "partition", &v);
|
||||
if (r == -ENOENT || /* Not a partition device */
|
||||
ERRNO_IS_PRIVILEGE(r)) /* Not ready to access? */
|
||||
if (sd_device_get_devtype(d, &v) < 0 || !streq(v, "partition"))
|
||||
return false;
|
||||
|
||||
r = sd_device_get_parent(d, &parent);
|
||||
if (r < 0)
|
||||
return false; /* Doesn't have a parent? No relevant to us */
|
||||
|
||||
r = sd_device_get_syspath(parent, &parent_syspath); /* Check parent of device of this action */
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_get_syspath(expected_parent, &expected_parent_syspath); /* Check parent of device we are looking for */
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (!path_equal(parent_syspath, expected_parent_syspath))
|
||||
return false; /* Has a different parent than what we need, not interesting to us */
|
||||
|
||||
r = sd_device_get_sysattr_value(d, "partition", &v);
|
||||
if (r < 0)
|
||||
return r;
|
||||
r = safe_atoi(v, &partno);
|
||||
@ -219,7 +244,7 @@ static int find_partition(
|
||||
return r;
|
||||
|
||||
FOREACH_DEVICE(e, q) {
|
||||
r = device_is_partition(q, pp);
|
||||
r = device_is_partition(q, parent, pp);
|
||||
if (r < 0)
|
||||
return r;
|
||||
if (r > 0) {
|
||||
@ -242,9 +267,7 @@ static inline void wait_data_done(struct wait_data *d) {
|
||||
}
|
||||
|
||||
static int device_monitor_handler(sd_device_monitor *monitor, sd_device *device, void *userdata) {
|
||||
const char *parent1_path, *parent2_path;
|
||||
struct wait_data *w = userdata;
|
||||
sd_device *pp;
|
||||
int r;
|
||||
|
||||
assert(w);
|
||||
@ -252,22 +275,7 @@ static int device_monitor_handler(sd_device_monitor *monitor, sd_device *device,
|
||||
if (device_for_action(device, SD_DEVICE_REMOVE))
|
||||
return 0;
|
||||
|
||||
r = sd_device_get_parent(device, &pp);
|
||||
if (r < 0)
|
||||
return 0; /* Doesn't have a parent? No relevant to us */
|
||||
|
||||
r = sd_device_get_syspath(pp, &parent1_path); /* Check parent of device of this action */
|
||||
if (r < 0)
|
||||
goto finish;
|
||||
|
||||
r = sd_device_get_syspath(w->parent_device, &parent2_path); /* Check parent of device we are looking for */
|
||||
if (r < 0)
|
||||
goto finish;
|
||||
|
||||
if (!path_equal(parent1_path, parent2_path))
|
||||
return 0; /* Has a different parent than what we need, not interesting to us */
|
||||
|
||||
r = device_is_partition(device, w->blkidp);
|
||||
r = device_is_partition(device, w->parent_device, w->blkidp);
|
||||
if (r < 0)
|
||||
goto finish;
|
||||
if (r == 0) /* Not the one we need */
|
||||
@ -313,6 +321,14 @@ static int wait_for_partition_device(
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_monitor_filter_add_match_parent(monitor, parent, true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_monitor_filter_add_match_sysattr(monitor, "partition", NULL, true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_device_monitor_attach_event(monitor, event);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
@ -136,6 +136,8 @@ int sd_device_monitor_stop(sd_device_monitor *m);
|
||||
|
||||
int sd_device_monitor_filter_add_match_subsystem_devtype(sd_device_monitor *m, const char *subsystem, const char *devtype);
|
||||
int sd_device_monitor_filter_add_match_tag(sd_device_monitor *m, const char *tag);
|
||||
int sd_device_monitor_filter_add_match_sysattr(sd_device_monitor *m, const char *sysattr, const char *value, int match);
|
||||
int sd_device_monitor_filter_add_match_parent(sd_device_monitor *m, sd_device *device, int match);
|
||||
int sd_device_monitor_filter_update(sd_device_monitor *m);
|
||||
int sd_device_monitor_filter_remove(sd_device_monitor *m);
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user