return 0;
}
+bool devpath_conflict(const char *a, const char *b) {
+ /* This returns true when two paths are equivalent, or one is a child of another. */
+
+ if (!a || !b)
+ return false;
+
+ for (; *a != '\0' && *b != '\0'; a++, b++)
+ if (*a != *b)
+ return false;
+
+ return *a == '/' || *b == '/' || *a == *b;
+}
+
int udev_queue_is_empty(void) {
return access("/run/udev/queue", F_OK) < 0 ?
(errno == ENOENT ? true : -errno) : false;
size_t udev_replace_chars(char *str, const char *allow);
int udev_resolve_subsys_kernel(const char *string, char *result, size_t maxsize, bool read_value);
+bool devpath_conflict(const char *a, const char *b);
+
int udev_queue_is_empty(void);
int udev_queue_init(void);
test_udev_resolve_subsys_kernel_one("[net/lo]/address", true, 0, "00:00:00:00:00:00");
}
+TEST(devpath_conflict) {
+ assert_se(!devpath_conflict(NULL, NULL));
+ assert_se(!devpath_conflict(NULL, "/devices/pci0000:00/0000:00:1c.4"));
+ assert_se(!devpath_conflict("/devices/pci0000:00/0000:00:1c.4", NULL));
+ assert_se(!devpath_conflict("/devices/pci0000:00/0000:00:1c.4", "/devices/pci0000:00/0000:00:00.0"));
+ assert_se(!devpath_conflict("/devices/virtual/net/veth99", "/devices/virtual/net/veth999"));
+
+ assert_se(devpath_conflict("/devices/pci0000:00/0000:00:1c.4", "/devices/pci0000:00/0000:00:1c.4"));
+ assert_se(devpath_conflict("/devices/pci0000:00/0000:00:1c.4", "/devices/pci0000:00/0000:00:1c.4/0000:3c:00.0"));
+ assert_se(devpath_conflict("/devices/pci0000:00/0000:00:1c.4/0000:3c:00.0/nvme/nvme0/nvme0n1",
+ "/devices/pci0000:00/0000:00:1c.4/0000:3c:00.0/nvme/nvme0/nvme0n1/nvme0n1p1"));
+}
+
DEFINE_TEST_MAIN(LOG_INFO);
static int event_is_blocked(Event *event) {
Event *loop_event = NULL;
- size_t devpath_len;
int r;
/* lookup event for identical, parent, child device */
assert(loop_event->seqnum > event->blocker_seqnum &&
loop_event->seqnum < event->seqnum);
- devpath_len = strlen(event->devpath);
-
/* check if queue contains events we depend on */
LIST_FOREACH(event, e, loop_event) {
- size_t loop_devpath_len, common;
-
loop_event = e;
/* found ourself, no later event can block us */
if (streq_ptr(loop_event->id, event->id))
break;
- /* check our old name */
- if (event->devpath_old && streq(event->devpath_old, loop_event->devpath))
- break;
-
- loop_devpath_len = strlen(loop_event->devpath);
-
- /* compare devpath */
- common = MIN(devpath_len, loop_devpath_len);
-
- /* one devpath is contained in the other? */
- if (!strneq(event->devpath, loop_event->devpath, common))
- continue;
-
- /* identical device event found */
- if (devpath_len == loop_devpath_len)
- break;
-
- /* parent device event found */
- if (event->devpath[common] == '/')
- break;
-
- /* child device event found */
- if (loop_event->devpath[common] == '/')
+ if (devpath_conflict(event->devpath, loop_event->devpath) ||
+ devpath_conflict(event->devpath, loop_event->devpath_old) ||
+ devpath_conflict(event->devpath_old, loop_event->devpath))
break;
}