ryantimwilson / rpms / systemd

Forked from rpms/systemd a month ago
Clone
Zbigniew Jędrzejewski-Szmek e4a83a
From 5b788e932fe918fb022bc20c3a15eb59e0fad53a Mon Sep 17 00:00:00 2001
Zbigniew Jędrzejewski-Szmek e4a83a
From: Lennart Poettering <lennart@poettering.net>
Zbigniew Jędrzejewski-Szmek e4a83a
Date: Fri, 27 Feb 2015 21:55:08 +0100
Zbigniew Jędrzejewski-Szmek e4a83a
Subject: [PATCH] core: rework device state logic
Zbigniew Jędrzejewski-Szmek e4a83a
Zbigniew Jędrzejewski-Szmek e4a83a
This change introduces a new state "tentative" for device units. Device
Zbigniew Jędrzejewski-Szmek e4a83a
units are considered "plugged" when udev announced them, "dead" when
Zbigniew Jędrzejewski-Szmek e4a83a
they are not available in the kernel, and "tentative" when they are
Zbigniew Jędrzejewski-Szmek e4a83a
referenced in /proc/self/mountinfo or /proc/swaps but not (yet)
Zbigniew Jędrzejewski-Szmek e4a83a
announced via udev.
Zbigniew Jędrzejewski-Szmek e4a83a
Zbigniew Jędrzejewski-Szmek e4a83a
This should fix a race when device nodes (like loop devices) are created
Zbigniew Jędrzejewski-Szmek e4a83a
and immediately mounted. Previously, systemd might end up seeing the
Zbigniew Jędrzejewski-Szmek e4a83a
mount unit before the device, and would thus pull down the mount because
Zbigniew Jędrzejewski-Szmek e4a83a
its BindTo dependency on the device would not be fulfilled.
Zbigniew Jędrzejewski-Szmek e4a83a
Zbigniew Jędrzejewski-Szmek e4a83a
(cherry picked from commit 628c89cc68ab96fce2de7ebba5933725d147aecc)
Zbigniew Jędrzejewski-Szmek e4a83a
---
Zbigniew Jędrzejewski-Szmek e4a83a
 src/core/device.c | 368 +++++++++++++++++++++++++++++++++---------------------
Zbigniew Jędrzejewski-Szmek e4a83a
 src/core/device.h |  14 ++-
Zbigniew Jędrzejewski-Szmek e4a83a
 src/core/mount.c  |  46 ++++---
Zbigniew Jędrzejewski-Szmek e4a83a
 src/core/swap.c   |  32 +++--
Zbigniew Jędrzejewski-Szmek e4a83a
 src/core/swap.h   |   4 +-
Zbigniew Jędrzejewski-Szmek e4a83a
 src/core/unit.c   |   1 -
Zbigniew Jędrzejewski-Szmek e4a83a
 6 files changed, 285 insertions(+), 180 deletions(-)
Zbigniew Jędrzejewski-Szmek e4a83a
Zbigniew Jędrzejewski-Szmek e4a83a
diff --git a/src/core/device.c b/src/core/device.c
Zbigniew Jędrzejewski-Szmek e4a83a
index d3deac3936..75b9a46287 100644
Zbigniew Jędrzejewski-Szmek e4a83a
--- a/src/core/device.c
Zbigniew Jędrzejewski-Szmek e4a83a
+++ b/src/core/device.c
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -36,7 +36,8 @@
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 static const UnitActiveState state_translation_table[_DEVICE_STATE_MAX] = {
Zbigniew Jędrzejewski-Szmek e4a83a
         [DEVICE_DEAD] = UNIT_INACTIVE,
Zbigniew Jędrzejewski-Szmek e4a83a
-        [DEVICE_PLUGGED] = UNIT_ACTIVE
Zbigniew Jędrzejewski-Szmek e4a83a
+        [DEVICE_TENTATIVE] = UNIT_ACTIVATING,
Zbigniew Jędrzejewski-Szmek e4a83a
+        [DEVICE_PLUGGED] = UNIT_ACTIVE,
Zbigniew Jędrzejewski-Szmek e4a83a
 };
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 static int device_dispatch_io(sd_event_source *source, int fd, uint32_t revents, void *userdata);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -65,6 +66,41 @@ static void device_unset_sysfs(Device *d) {
Zbigniew Jędrzejewski-Szmek e4a83a
         d->sysfs = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
+static int device_set_sysfs(Device *d, const char *sysfs) {
Zbigniew Jędrzejewski-Szmek e4a83a
+        Device *first;
Zbigniew Jędrzejewski-Szmek e4a83a
+        char *copy;
Zbigniew Jędrzejewski-Szmek e4a83a
+        int r;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        assert(d);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (streq_ptr(d->sysfs, sysfs))
Zbigniew Jędrzejewski-Szmek e4a83a
+                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        r = hashmap_ensure_allocated(&UNIT(d)->manager->devices_by_sysfs, &string_hash_ops);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return r;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        copy = strdup(sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (!copy)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        device_unset_sysfs(d);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        first = hashmap_get(UNIT(d)->manager->devices_by_sysfs, sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
+        LIST_PREPEND(same_sysfs, first, d);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        r = hashmap_replace(UNIT(d)->manager->devices_by_sysfs, copy, first);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (r < 0) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                LIST_REMOVE(same_sysfs, first, d);
Zbigniew Jędrzejewski-Szmek e4a83a
+                free(copy);
Zbigniew Jędrzejewski-Szmek e4a83a
+                return r;
Zbigniew Jędrzejewski-Szmek e4a83a
+        }
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        d->sysfs = copy;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+}
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
 static void device_init(Unit *u) {
Zbigniew Jędrzejewski-Szmek e4a83a
         Device *d = DEVICE(u);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -112,8 +148,13 @@ static int device_coldplug(Unit *u) {
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(d);
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(d->state == DEVICE_DEAD);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (d->sysfs)
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (d->found & DEVICE_FOUND_UDEV)
Zbigniew Jędrzejewski-Szmek e4a83a
+                /* If udev says the device is around, it's around */
Zbigniew Jędrzejewski-Szmek e4a83a
                 device_set_state(d, DEVICE_PLUGGED);
Zbigniew Jędrzejewski-Szmek e4a83a
+        else if (d->found != DEVICE_NOT_FOUND)
Zbigniew Jędrzejewski-Szmek e4a83a
+                /* If a device is found in /proc/self/mountinfo or
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * /proc/swaps, it's "tentatively" around. */
Zbigniew Jędrzejewski-Szmek e4a83a
+                device_set_state(d, DEVICE_TENTATIVE);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -142,49 +183,9 @@ _pure_ static const char *device_sub_state_to_string(Unit *u) {
Zbigniew Jędrzejewski-Szmek e4a83a
         return device_state_to_string(DEVICE(u)->state);
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int device_add_escaped_name(Unit *u, const char *dn) {
Zbigniew Jędrzejewski-Szmek e4a83a
-        _cleanup_free_ char *e = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
-        int r;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(u);
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(dn);
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(dn[0] == '/');
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        e = unit_name_from_path(dn, ".device");
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!e)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        r = unit_add_name(u, e);
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (r < 0 && r != -EEXIST)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return r;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
-}
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-static int device_find_escape_name(Manager *m, const char *dn, Unit **_u) {
Zbigniew Jędrzejewski-Szmek e4a83a
-        _cleanup_free_ char *e = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
-        Unit *u;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(dn);
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(dn[0] == '/');
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(_u);
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        e = unit_name_from_path(dn, ".device");
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!e)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        u = manager_get_unit(m, e);
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (u) {
Zbigniew Jędrzejewski-Szmek e4a83a
-                *_u = u;
Zbigniew Jędrzejewski-Szmek e4a83a
-                return 1;
Zbigniew Jędrzejewski-Szmek e4a83a
-        }
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
-}
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-static int device_make_description(Unit *u, struct udev_device *dev, const char *path) {
Zbigniew Jędrzejewski-Szmek e4a83a
+static int device_update_description(Unit *u, struct udev_device *dev, const char *path) {
Zbigniew Jędrzejewski-Szmek e4a83a
         const char *model;
Zbigniew Jędrzejewski-Szmek e4a83a
+        int r;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(u);
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -209,13 +210,16 @@ static int device_make_description(Unit *u, struct udev_device *dev, const char
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                         j = strjoin(model, " ", label, NULL);
Zbigniew Jędrzejewski-Szmek e4a83a
                         if (j)
Zbigniew Jędrzejewski-Szmek e4a83a
-                                return unit_set_description(u, j);
Zbigniew Jędrzejewski-Szmek e4a83a
-                }
Zbigniew Jędrzejewski-Szmek e4a83a
+                                r = unit_set_description(u, j);
Zbigniew Jędrzejewski-Szmek e4a83a
+                } else
Zbigniew Jędrzejewski-Szmek e4a83a
+                        r = unit_set_description(u, model);
Zbigniew Jędrzejewski-Szmek e4a83a
+        } else
Zbigniew Jędrzejewski-Szmek e4a83a
+                r = unit_set_description(u, path);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                return unit_set_description(u, model);
Zbigniew Jędrzejewski-Szmek e4a83a
-        }
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
+                log_unit_error_errno(u->id, r, "Failed to set device description: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        return unit_set_description(u, path);
Zbigniew Jędrzejewski-Szmek e4a83a
+        return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 static int device_add_udev_wants(Unit *u, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -242,20 +246,20 @@ static int device_add_udev_wants(Unit *u, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 n = unit_name_mangle(e, MANGLE_NOGLOB);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (!n)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 r = unit_add_dependency_by_name(u, UNIT_WANTS, n, NULL, true);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        return r;
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_unit_error_errno(u->id, r, "Failed to add wants dependency: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
         }
Zbigniew Jędrzejewski-Szmek e4a83a
         if (!isempty(state))
Zbigniew Jędrzejewski-Szmek e4a83a
-                log_unit_warning(u->id, "Property %s on %s has trailing garbage, ignoring.",
Zbigniew Jędrzejewski-Szmek e4a83a
-                                 property, strna(udev_device_get_syspath(dev)));
Zbigniew Jędrzejewski-Szmek e4a83a
+                log_unit_warning(u->id, "Property %s on %s has trailing garbage, ignoring.", property, strna(udev_device_get_syspath(dev)));
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int device_update_unit(Manager *m, struct udev_device *dev, const char *path, bool main) {
Zbigniew Jędrzejewski-Szmek e4a83a
+static int device_setup_unit(Manager *m, struct udev_device *dev, const char *path, bool main) {
Zbigniew Jędrzejewski-Szmek e4a83a
+        _cleanup_free_ char *e = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         const char *sysfs;
Zbigniew Jędrzejewski-Szmek e4a83a
         Unit *u = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         bool delete;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -269,12 +273,18 @@ static int device_update_unit(Manager *m, struct udev_device *dev, const char *p
Zbigniew Jędrzejewski-Szmek e4a83a
         if (!sysfs)
Zbigniew Jędrzejewski-Szmek e4a83a
                 return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        r = device_find_escape_name(m, path, &u);
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return r;
Zbigniew Jędrzejewski-Szmek e4a83a
+        e = unit_name_from_path(path, ".device");
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (!e)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        u = manager_get_unit(m, e);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (u && DEVICE(u)->sysfs && !path_equal(DEVICE(u)->sysfs, sysfs))
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (u &&
Zbigniew Jędrzejewski-Szmek e4a83a
+            DEVICE(u)->sysfs &&
Zbigniew Jędrzejewski-Szmek e4a83a
+            !path_equal(DEVICE(u)->sysfs, sysfs)) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                log_unit_error(u->id, "Device %s appeared twice with different sysfs paths %s and %s", e, DEVICE(u)->sysfs, sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
                 return -EEXIST;
Zbigniew Jędrzejewski-Szmek e4a83a
+        }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         if (!u) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 delete = true;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -283,7 +293,7 @@ static int device_update_unit(Manager *m, struct udev_device *dev, const char *p
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (!u)
Zbigniew Jędrzejewski-Szmek e4a83a
                         return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = device_add_escaped_name(u, path);
Zbigniew Jędrzejewski-Szmek e4a83a
+                r = unit_add_name(u, e);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                         goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -295,37 +305,16 @@ static int device_update_unit(Manager *m, struct udev_device *dev, const char *p
Zbigniew Jędrzejewski-Szmek e4a83a
          * actually been seen yet ->sysfs will not be
Zbigniew Jędrzejewski-Szmek e4a83a
          * initialized. Hence initialize it if necessary. */
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!DEVICE(u)->sysfs) {
Zbigniew Jędrzejewski-Szmek e4a83a
-                Device *first;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-                DEVICE(u)->sysfs = strdup(sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
-                if (!DEVICE(u)->sysfs) {
Zbigniew Jędrzejewski-Szmek e4a83a
-                        r = -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
-                        goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
-                }
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = hashmap_ensure_allocated(&m->devices_by_sysfs, &string_hash_ops);
Zbigniew Jędrzejewski-Szmek e4a83a
-                if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-                first = hashmap_get(m->devices_by_sysfs, sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
-                LIST_PREPEND(same_sysfs, first, DEVICE(u));
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = hashmap_replace(m->devices_by_sysfs, DEVICE(u)->sysfs, first);
Zbigniew Jędrzejewski-Szmek e4a83a
-                if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
-        }
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        device_make_description(u, dev, path);
Zbigniew Jędrzejewski-Szmek e4a83a
+        r = device_set_sysfs(DEVICE(u), sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
+                goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (main) {
Zbigniew Jędrzejewski-Szmek e4a83a
-                /* The additional systemd udev properties we only
Zbigniew Jędrzejewski-Szmek e4a83a
-                 * interpret for the main object */
Zbigniew Jędrzejewski-Szmek e4a83a
+        (void) device_update_description(u, dev, path);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = device_add_udev_wants(u, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
-                if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
-        }
Zbigniew Jędrzejewski-Szmek e4a83a
+        /* The additional systemd udev properties we only interpret
Zbigniew Jędrzejewski-Szmek e4a83a
+         * for the main object */
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (main)
Zbigniew Jędrzejewski-Szmek e4a83a
+                (void) device_add_udev_wants(u, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         /* Note that this won't dispatch the load queue, the caller
Zbigniew Jędrzejewski-Szmek e4a83a
          * has to do that if needed and appropriate */
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -334,7 +323,7 @@ static int device_update_unit(Manager *m, struct udev_device *dev, const char *p
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 fail:
Zbigniew Jędrzejewski-Szmek e4a83a
-        log_warning_errno(r, "Failed to load device unit: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
+        log_unit_warning_errno(u->id, r, "Failed to set up device unit: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         if (delete && u)
Zbigniew Jędrzejewski-Szmek e4a83a
                 unit_free(u);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -342,7 +331,7 @@ fail:
Zbigniew Jędrzejewski-Szmek e4a83a
         return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int device_process_new_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
+static int device_process_new(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
         const char *sysfs, *dn, *alias;
Zbigniew Jędrzejewski-Szmek e4a83a
         struct udev_list_entry *item = NULL, *first = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         int r;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -354,14 +343,14 @@ static int device_process_new_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         /* Add the main unit named after the sysfs path */
Zbigniew Jędrzejewski-Szmek e4a83a
-        r = device_update_unit(m, dev, sysfs, true);
Zbigniew Jędrzejewski-Szmek e4a83a
+        r = device_setup_unit(m, dev, sysfs, true);
Zbigniew Jędrzejewski-Szmek e4a83a
         if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                 return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         /* Add an additional unit for the device node */
Zbigniew Jędrzejewski-Szmek e4a83a
         dn = udev_device_get_devnode(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
         if (dn)
Zbigniew Jędrzejewski-Szmek e4a83a
-                device_update_unit(m, dev, dn, false);
Zbigniew Jędrzejewski-Szmek e4a83a
+                (void) device_setup_unit(m, dev, dn, false);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         /* Add additional units for all symlinks */
Zbigniew Jędrzejewski-Szmek e4a83a
         first = udev_device_get_devlinks_list_entry(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -388,7 +377,7 @@ static int device_process_new_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
                             st.st_rdev != udev_device_get_devnum(dev))
Zbigniew Jędrzejewski-Szmek e4a83a
                                 continue;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                device_update_unit(m, dev, p, false);
Zbigniew Jędrzejewski-Szmek e4a83a
+                (void) device_setup_unit(m, dev, p, false);
Zbigniew Jędrzejewski-Szmek e4a83a
         }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         /* Add additional units for all explicitly configured
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -405,7 +394,7 @@ static int device_process_new_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
                         e[l] = 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                         if (path_is_absolute(e))
Zbigniew Jędrzejewski-Szmek e4a83a
-                                device_update_unit(m, dev, e, false);
Zbigniew Jędrzejewski-Szmek e4a83a
+                                (void) device_setup_unit(m, dev, e, false);
Zbigniew Jędrzejewski-Szmek e4a83a
                         else
Zbigniew Jędrzejewski-Szmek e4a83a
                                 log_warning("SYSTEMD_ALIAS for %s is not an absolute path, ignoring: %s", sysfs, e);
Zbigniew Jędrzejewski-Szmek e4a83a
                 }
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -416,39 +405,62 @@ static int device_process_new_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static void device_set_path_plugged(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
-        const char *sysfs;
Zbigniew Jędrzejewski-Szmek e4a83a
+static void device_update_found_one(Device *d, bool add, DeviceFound found, bool now) {
Zbigniew Jędrzejewski-Szmek e4a83a
+        DeviceFound n;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        assert(d);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        n = add ? (d->found | found) : (d->found & ~found);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (n == d->found)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        d->found = n;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (now) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (d->found & DEVICE_FOUND_UDEV)
Zbigniew Jędrzejewski-Szmek e4a83a
+                        device_set_state(d, DEVICE_PLUGGED);
Zbigniew Jędrzejewski-Szmek e4a83a
+                else if (d->found != DEVICE_NOT_FOUND)
Zbigniew Jędrzejewski-Szmek e4a83a
+                        device_set_state(d, DEVICE_TENTATIVE);
Zbigniew Jędrzejewski-Szmek e4a83a
+                else
Zbigniew Jędrzejewski-Szmek e4a83a
+                        device_set_state(d, DEVICE_DEAD);
Zbigniew Jędrzejewski-Szmek e4a83a
+        }
Zbigniew Jędrzejewski-Szmek e4a83a
+}
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+static int device_update_found_by_sysfs(Manager *m, const char *sysfs, bool add, DeviceFound found, bool now) {
Zbigniew Jędrzejewski-Szmek e4a83a
         Device *d, *l;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+        assert(sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        sysfs = udev_device_get_syspath(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!sysfs)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return;
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (found == DEVICE_NOT_FOUND)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         l = hashmap_get(m->devices_by_sysfs, sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
         LIST_FOREACH(same_sysfs, d, l)
Zbigniew Jędrzejewski-Szmek e4a83a
-                device_set_state(d, DEVICE_PLUGGED);
Zbigniew Jędrzejewski-Szmek e4a83a
+                device_update_found_one(d, add, found, now);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int device_process_removed_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
-        const char *sysfs;
Zbigniew Jędrzejewski-Szmek e4a83a
-        Device *d;
Zbigniew Jędrzejewski-Szmek e4a83a
+static int device_update_found_by_name(Manager *m, const char *path, bool add, DeviceFound found, bool now) {
Zbigniew Jędrzejewski-Szmek e4a83a
+        _cleanup_free_ char *e = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
+        Unit *u;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+        assert(path);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        sysfs = udev_device_get_syspath(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!sysfs)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (found == DEVICE_NOT_FOUND)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        /* Remove all units of this sysfs path */
Zbigniew Jędrzejewski-Szmek e4a83a
-        while ((d = hashmap_get(m->devices_by_sysfs, sysfs))) {
Zbigniew Jędrzejewski-Szmek e4a83a
-                device_unset_sysfs(d);
Zbigniew Jędrzejewski-Szmek e4a83a
-                device_set_state(d, DEVICE_DEAD);
Zbigniew Jędrzejewski-Szmek e4a83a
-        }
Zbigniew Jędrzejewski-Szmek e4a83a
+        e = unit_name_from_path(path, ".device");
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (!e)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
+        u = manager_get_unit(m, e);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (!u)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        device_update_found_one(DEVICE(u), add, found, now);
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -464,22 +476,6 @@ static bool device_is_ready(struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
         return parse_boolean(ready) != 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int device_process_new_path(Manager *m, const char *path) {
Zbigniew Jędrzejewski-Szmek e4a83a
-        _cleanup_udev_device_unref_ struct udev_device *dev = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
-        assert(path);
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        dev = udev_device_new_from_syspath(m->udev, path);
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!dev)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!device_is_ready(dev))
Zbigniew Jędrzejewski-Szmek e4a83a
-                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-        return device_process_new_device(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
-}
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
 static Unit *device_following(Unit *u) {
Zbigniew Jędrzejewski-Szmek e4a83a
         Device *d = DEVICE(u);
Zbigniew Jędrzejewski-Szmek e4a83a
         Device *other, *first = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -606,12 +602,31 @@ static int device_enumerate(Manager *m) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         first = udev_enumerate_get_list_entry(e);
Zbigniew Jędrzejewski-Szmek e4a83a
-        udev_list_entry_foreach(item, first)
Zbigniew Jędrzejewski-Szmek e4a83a
-                device_process_new_path(m, udev_list_entry_get_name(item));
Zbigniew Jędrzejewski-Szmek e4a83a
+        udev_list_entry_foreach(item, first) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                _cleanup_udev_device_unref_ struct udev_device *dev = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
+                const char *sysfs;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                sysfs = udev_list_entry_get_name(item);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                dev = udev_device_new_from_syspath(m->udev, sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (!dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                        log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
+                        continue;
Zbigniew Jędrzejewski-Szmek e4a83a
+                }
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (!device_is_ready(dev))
Zbigniew Jędrzejewski-Szmek e4a83a
+                        continue;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                (void) device_process_new(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                device_update_found_by_sysfs(m, sysfs, true, DEVICE_FOUND_UDEV, false);
Zbigniew Jędrzejewski-Szmek e4a83a
+        }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 fail:
Zbigniew Jędrzejewski-Szmek e4a83a
+        log_error_errno(r, "Failed to enumerate devices: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
         device_shutdown(m);
Zbigniew Jędrzejewski-Szmek e4a83a
         return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -619,7 +634,7 @@ fail:
Zbigniew Jędrzejewski-Szmek e4a83a
 static int device_dispatch_io(sd_event_source *source, int fd, uint32_t revents, void *userdata) {
Zbigniew Jędrzejewski-Szmek e4a83a
         _cleanup_udev_device_unref_ struct udev_device *dev = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         Manager *m = userdata;
Zbigniew Jędrzejewski-Szmek e4a83a
-        const char *action;
Zbigniew Jędrzejewski-Szmek e4a83a
+        const char *action, *sysfs;
Zbigniew Jędrzejewski-Szmek e4a83a
         int r;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -641,33 +656,47 @@ static int device_dispatch_io(sd_event_source *source, int fd, uint32_t revents,
Zbigniew Jędrzejewski-Szmek e4a83a
         if (!dev)
Zbigniew Jędrzejewski-Szmek e4a83a
                 return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
+        sysfs = udev_device_get_syspath(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (!sysfs) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                log_error("Failed to get udev sys path.");
Zbigniew Jędrzejewski-Szmek e4a83a
+                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+        }
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
         action = udev_device_get_action(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
         if (!action) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 log_error("Failed to get udev action string.");
Zbigniew Jędrzejewski-Szmek e4a83a
                 return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
         }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (streq(action, "remove") || !device_is_ready(dev))  {
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = device_process_removed_device(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
-                if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        log_error_errno(r, "Failed to process device remove event: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = swap_process_removed_device(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (streq(action, "remove"))  {
Zbigniew Jędrzejewski-Szmek e4a83a
+                r = swap_process_device_remove(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                         log_error_errno(r, "Failed to process swap device remove event: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        } else {
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = device_process_new_device(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
-                if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        log_error_errno(r, "Failed to process device new event: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
+                /* If we get notified that a device was removed by
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * udev, then it's completely gone, hence unset all
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * found bits */
Zbigniew Jędrzejewski-Szmek e4a83a
+                device_update_found_by_sysfs(m, sysfs, false, DEVICE_FOUND_UDEV|DEVICE_FOUND_MOUNT|DEVICE_FOUND_SWAP, true);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                r = swap_process_new_device(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+        } else if (device_is_ready(dev)) {
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                (void) device_process_new(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                r = swap_process_device_new(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                         log_error_errno(r, "Failed to process swap device new event: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 manager_dispatch_load_queue(m);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                device_set_path_plugged(m, dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+                /* The device is found now, set the udev found bit */
Zbigniew Jędrzejewski-Szmek e4a83a
+                device_update_found_by_sysfs(m, sysfs, true, DEVICE_FOUND_UDEV, true);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        } else {
Zbigniew Jędrzejewski-Szmek e4a83a
+                /* The device is nominally around, but not ready for
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * us. Hence unset the udev bit, but leave the rest
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * around. */
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                device_update_found_by_sysfs(m, sysfs, false, DEVICE_FOUND_UDEV, true);
Zbigniew Jędrzejewski-Szmek e4a83a
         }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -686,9 +715,58 @@ static bool device_supported(Manager *m) {
Zbigniew Jędrzejewski-Szmek e4a83a
         return read_only <= 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
+int device_found_node(Manager *m, const char *node, bool add, DeviceFound found, bool now) {
Zbigniew Jędrzejewski-Szmek e4a83a
+        _cleanup_udev_device_unref_ struct udev_device *dev = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
+        struct stat st;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
+        assert(node);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        /* This is called whenever we find a device referenced in
Zbigniew Jędrzejewski-Szmek e4a83a
+         * /proc/swaps or /proc/self/mounts. Such a device might be
Zbigniew Jędrzejewski-Szmek e4a83a
+         * mounted/enabled at a time where udev has not finished
Zbigniew Jędrzejewski-Szmek e4a83a
+         * probing it yet, and we thus haven't learned about it
Zbigniew Jędrzejewski-Szmek e4a83a
+         * yet. In this case we will set the device unit to
Zbigniew Jędrzejewski-Szmek e4a83a
+         * "tentative" state. */
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (add) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (!path_startswith(node, "/dev"))
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (stat(node, &st) < 0) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                        if (errno == ENOENT)
Zbigniew Jędrzejewski-Szmek e4a83a
+                                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_error_errno(errno, "Failed to stat device node file %s: %m", node);
Zbigniew Jędrzejewski-Szmek e4a83a
+                }
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (!S_ISBLK(st.st_mode) && !S_ISCHR(st.st_mode))
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                dev = udev_device_new_from_devnum(m->udev, S_ISBLK(st.st_mode) ? 'b' : 'c', st.st_rdev);
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (!dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                        if (errno == ENOENT)
Zbigniew Jędrzejewski-Szmek e4a83a
+                                return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
+                }
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                /* If the device is known in the kernel and newly
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * appeared, then we'll create a device unit for it,
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * under the name referenced in /proc/swaps or
Zbigniew Jędrzejewski-Szmek e4a83a
+                 * /proc/self/mountinfo. */
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                (void) device_setup_unit(m, dev, node, false);
Zbigniew Jędrzejewski-Szmek e4a83a
+        }
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        /* Update the device unit's state, should it exist */
Zbigniew Jędrzejewski-Szmek e4a83a
+        return device_update_found_by_name(m, node, add, found, now);
Zbigniew Jędrzejewski-Szmek e4a83a
+}
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
 static const char* const device_state_table[_DEVICE_STATE_MAX] = {
Zbigniew Jędrzejewski-Szmek e4a83a
         [DEVICE_DEAD] = "dead",
Zbigniew Jędrzejewski-Szmek e4a83a
-        [DEVICE_PLUGGED] = "plugged"
Zbigniew Jędrzejewski-Szmek e4a83a
+        [DEVICE_TENTATIVE] = "tentative",
Zbigniew Jędrzejewski-Szmek e4a83a
+        [DEVICE_PLUGGED] = "plugged",
Zbigniew Jędrzejewski-Szmek e4a83a
 };
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 DEFINE_STRING_TABLE_LOOKUP(device_state, DeviceState);
Zbigniew Jędrzejewski-Szmek e4a83a
diff --git a/src/core/device.h b/src/core/device.h
Zbigniew Jędrzejewski-Szmek e4a83a
index bb7ae07834..0609b20fdb 100644
Zbigniew Jędrzejewski-Szmek e4a83a
--- a/src/core/device.h
Zbigniew Jędrzejewski-Szmek e4a83a
+++ b/src/core/device.h
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -29,20 +29,28 @@ typedef struct Device Device;
Zbigniew Jędrzejewski-Szmek e4a83a
  * simplifies the state engine greatly */
Zbigniew Jędrzejewski-Szmek e4a83a
 typedef enum DeviceState {
Zbigniew Jędrzejewski-Szmek e4a83a
         DEVICE_DEAD,
Zbigniew Jędrzejewski-Szmek e4a83a
-        DEVICE_PLUGGED,
Zbigniew Jędrzejewski-Szmek e4a83a
+        DEVICE_TENTATIVE, /* mounted or swapped, but not (yet) announced by udev */
Zbigniew Jędrzejewski-Szmek e4a83a
+        DEVICE_PLUGGED,   /* announced by udev */
Zbigniew Jędrzejewski-Szmek e4a83a
         _DEVICE_STATE_MAX,
Zbigniew Jędrzejewski-Szmek e4a83a
         _DEVICE_STATE_INVALID = -1
Zbigniew Jędrzejewski-Szmek e4a83a
 } DeviceState;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
+typedef enum DeviceFound {
Zbigniew Jędrzejewski-Szmek e4a83a
+        DEVICE_NOT_FOUND = 0,
Zbigniew Jędrzejewski-Szmek e4a83a
+        DEVICE_FOUND_UDEV = 1,
Zbigniew Jędrzejewski-Szmek e4a83a
+        DEVICE_FOUND_MOUNT = 2,
Zbigniew Jędrzejewski-Szmek e4a83a
+        DEVICE_FOUND_SWAP = 4,
Zbigniew Jędrzejewski-Szmek e4a83a
+} DeviceFound;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
 struct Device {
Zbigniew Jędrzejewski-Szmek e4a83a
         Unit meta;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         char *sysfs;
Zbigniew Jędrzejewski-Szmek e4a83a
+        DeviceFound found;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         /* In order to be able to distinguish dependencies on
Zbigniew Jędrzejewski-Szmek e4a83a
         different device nodes we might end up creating multiple
Zbigniew Jędrzejewski-Szmek e4a83a
         devices for the same sysfs path. We chain them up here. */
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
         LIST_FIELDS(struct Device, same_sysfs);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         DeviceState state;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -52,3 +60,5 @@ extern const UnitVTable device_vtable;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 const char* device_state_to_string(DeviceState i) _const_;
Zbigniew Jędrzejewski-Szmek e4a83a
 DeviceState device_state_from_string(const char *s) _pure_;
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+int device_found_node(Manager *m, const char *node, bool add, DeviceFound found, bool now);
Zbigniew Jędrzejewski-Szmek e4a83a
diff --git a/src/core/mount.c b/src/core/mount.c
Zbigniew Jędrzejewski-Szmek e4a83a
index f3977e62de..c971330af2 100644
Zbigniew Jędrzejewski-Szmek e4a83a
--- a/src/core/mount.c
Zbigniew Jędrzejewski-Szmek e4a83a
+++ b/src/core/mount.c
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1391,7 +1391,7 @@ static int mount_dispatch_timer(sd_event_source *source, usec_t usec, void *user
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int mount_add_one(
Zbigniew Jędrzejewski-Szmek e4a83a
+static int mount_setup_unit(
Zbigniew Jędrzejewski-Szmek e4a83a
                 Manager *m,
Zbigniew Jędrzejewski-Szmek e4a83a
                 const char *what,
Zbigniew Jędrzejewski-Szmek e4a83a
                 const char *where,
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1434,7 +1434,7 @@ static int mount_add_one(
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 u = unit_new(m, sizeof(Mount));
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (!u)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 r = unit_add_name(u, e);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1547,6 +1547,8 @@ static int mount_add_one(
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 fail:
Zbigniew Jędrzejewski-Szmek e4a83a
+        log_warning_errno(r, "Failed to set up mount unit: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
         if (delete && u)
Zbigniew Jędrzejewski-Szmek e4a83a
                 unit_free(u);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1554,33 +1556,36 @@ fail:
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 static int mount_load_proc_self_mountinfo(Manager *m, bool set_flags) {
Zbigniew Jędrzejewski-Szmek e4a83a
-        _cleanup_(mnt_free_tablep) struct libmnt_table *tb = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
-        _cleanup_(mnt_free_iterp) struct libmnt_iter *itr = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
-        struct libmnt_fs *fs;
Zbigniew Jędrzejewski-Szmek e4a83a
+        _cleanup_(mnt_free_tablep) struct libmnt_table *t = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
+        _cleanup_(mnt_free_iterp) struct libmnt_iter *i = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         int r = 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        tb = mnt_new_table();
Zbigniew Jędrzejewski-Szmek e4a83a
-        itr = mnt_new_iter(MNT_ITER_FORWARD);
Zbigniew Jędrzejewski-Szmek e4a83a
-        if (!tb || !itr)
Zbigniew Jędrzejewski-Szmek e4a83a
+        t = mnt_new_table();
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (!t)
Zbigniew Jędrzejewski-Szmek e4a83a
                 return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        r = mnt_table_parse_mtab(tb, NULL);
Zbigniew Jędrzejewski-Szmek e4a83a
+        i = mnt_new_iter(MNT_ITER_FORWARD);
Zbigniew Jędrzejewski-Szmek e4a83a
+        if (!i)
Zbigniew Jędrzejewski-Szmek e4a83a
+                return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+        r = mnt_table_parse_mtab(t, NULL);
Zbigniew Jędrzejewski-Szmek e4a83a
         if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                return r;
Zbigniew Jędrzejewski-Szmek e4a83a
+                return log_error_errno(r, "Failed to parse /proc/self/mountinfo: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         r = 0;
Zbigniew Jędrzejewski-Szmek e4a83a
         for (;;) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 const char *device, *path, *options, *fstype;
Zbigniew Jędrzejewski-Szmek e4a83a
                 _cleanup_free_ const char *d = NULL, *p = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
+                struct libmnt_fs *fs;
Zbigniew Jędrzejewski-Szmek e4a83a
                 int k;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                k = mnt_table_next_fs(tb, itr, &fs);
Zbigniew Jędrzejewski-Szmek e4a83a
+                k = mnt_table_next_fs(t, i, &fs);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (k == 1)
Zbigniew Jędrzejewski-Szmek e4a83a
                         break;
Zbigniew Jędrzejewski-Szmek e4a83a
-                else if (k < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        return log_error_errno(k, "Failed to get next entry from /etc/fstab: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (k < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_error_errno(k, "Failed to get next entry from /proc/self/mountinfo: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 device = mnt_fs_get_source(fs);
Zbigniew Jędrzejewski-Szmek e4a83a
                 path = mnt_fs_get_target(fs);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1588,11 +1593,16 @@ static int mount_load_proc_self_mountinfo(Manager *m, bool set_flags) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 fstype = mnt_fs_get_fstype(fs);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 d = cunescape(device);
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (!d)
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
                 p = cunescape(path);
Zbigniew Jędrzejewski-Szmek e4a83a
-                if (!d || !p)
Zbigniew Jędrzejewski-Szmek e4a83a
+                if (!p)
Zbigniew Jędrzejewski-Szmek e4a83a
                         return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                k = mount_add_one(m, d, p, options, fstype, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
+                (void) device_found_node(m, d, true, DEVICE_FOUND_MOUNT, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                k = mount_setup_unit(m, d, p, options, fstype, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (r == 0 && k < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                         r = k;
Zbigniew Jędrzejewski-Szmek e4a83a
         }
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1736,8 +1746,6 @@ static int mount_dispatch_io(sd_event_source *source, int fd, uint32_t revents,
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         r = mount_load_proc_self_mountinfo(m, true);
Zbigniew Jędrzejewski-Szmek e4a83a
         if (r < 0) {
Zbigniew Jędrzejewski-Szmek e4a83a
-                log_error_errno(r, "Failed to reread /proc/self/mountinfo: %m");
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
                 /* Reset flags, just in case, for later calls */
Zbigniew Jędrzejewski-Szmek e4a83a
                 LIST_FOREACH(units_by_type, u, m->units_by_type[UNIT_MOUNT]) {
Zbigniew Jędrzejewski-Szmek e4a83a
                         Mount *mount = MOUNT(u);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1770,6 +1778,10 @@ static int mount_dispatch_io(sd_event_source *source, int fd, uint32_t revents,
Zbigniew Jędrzejewski-Szmek e4a83a
                                 break;
Zbigniew Jędrzejewski-Szmek e4a83a
                         }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
+                        if (mount->parameters_proc_self_mountinfo.what)
Zbigniew Jędrzejewski-Szmek e4a83a
+                                (void) device_found_node(m, mount->parameters_proc_self_mountinfo.what, false, DEVICE_FOUND_MOUNT, true);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
                 } else if (mount->just_mounted || mount->just_changed) {
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                         /* New or changed mount entry */
Zbigniew Jędrzejewski-Szmek e4a83a
diff --git a/src/core/swap.c b/src/core/swap.c
Zbigniew Jędrzejewski-Szmek e4a83a
index 6997921fde..5c19af5d91 100644
Zbigniew Jędrzejewski-Szmek e4a83a
--- a/src/core/swap.c
Zbigniew Jędrzejewski-Szmek e4a83a
+++ b/src/core/swap.c
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -338,7 +338,7 @@ static int swap_load(Unit *u) {
Zbigniew Jędrzejewski-Szmek e4a83a
         return swap_verify(s);
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int swap_add_one(
Zbigniew Jędrzejewski-Szmek e4a83a
+static int swap_setup_unit(
Zbigniew Jędrzejewski-Szmek e4a83a
                 Manager *m,
Zbigniew Jędrzejewski-Szmek e4a83a
                 const char *what,
Zbigniew Jędrzejewski-Szmek e4a83a
                 const char *what_proc_swaps,
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -363,8 +363,10 @@ static int swap_add_one(
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         if (u &&
Zbigniew Jędrzejewski-Szmek e4a83a
             SWAP(u)->from_proc_swaps &&
Zbigniew Jędrzejewski-Szmek e4a83a
-            !path_equal(SWAP(u)->parameters_proc_swaps.what, what_proc_swaps))
Zbigniew Jędrzejewski-Szmek e4a83a
+            !path_equal(SWAP(u)->parameters_proc_swaps.what, what_proc_swaps)) {
Zbigniew Jędrzejewski-Szmek e4a83a
+                log_error("Swap %s appeared twice with different device paths %s and %s", e, SWAP(u)->parameters_proc_swaps.what, what_proc_swaps);
Zbigniew Jędrzejewski-Szmek e4a83a
                 return -EEXIST;
Zbigniew Jędrzejewski-Szmek e4a83a
+        }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         if (!u) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 delete = true;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -379,7 +381,7 @@ static int swap_add_one(
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 SWAP(u)->what = strdup(what);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (!SWAP(u)->what) {
Zbigniew Jędrzejewski-Szmek e4a83a
-                        r = log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
+                        r = -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
                         goto fail;
Zbigniew Jędrzejewski-Szmek e4a83a
                 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -407,7 +409,6 @@ static int swap_add_one(
Zbigniew Jędrzejewski-Szmek e4a83a
         p->priority = priority;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         unit_add_to_dbus_queue(u);
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
         return 0;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 fail:
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -419,7 +420,7 @@ fail:
Zbigniew Jędrzejewski-Szmek e4a83a
         return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-static int swap_process_new_swap(Manager *m, const char *device, int prio, bool set_flags) {
Zbigniew Jędrzejewski-Szmek e4a83a
+static int swap_process_new(Manager *m, const char *device, int prio, bool set_flags) {
Zbigniew Jędrzejewski-Szmek e4a83a
         _cleanup_udev_device_unref_ struct udev_device *d = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         struct udev_list_entry *item = NULL, *first = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         const char *dn;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -428,7 +429,7 @@ static int swap_process_new_swap(Manager *m, const char *device, int prio, bool
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         assert(m);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-        r = swap_add_one(m, device, device, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
+        r = swap_setup_unit(m, device, device, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
         if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                 return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -444,7 +445,7 @@ static int swap_process_new_swap(Manager *m, const char *device, int prio, bool
Zbigniew Jędrzejewski-Szmek e4a83a
         /* Add the main device node */
Zbigniew Jędrzejewski-Szmek e4a83a
         dn = udev_device_get_devnode(d);
Zbigniew Jędrzejewski-Szmek e4a83a
         if (dn && !streq(dn, device))
Zbigniew Jędrzejewski-Szmek e4a83a
-                swap_add_one(m, dn, device, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
+                swap_setup_unit(m, dn, device, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         /* Add additional units for all symlinks */
Zbigniew Jędrzejewski-Szmek e4a83a
         first = udev_device_get_devlinks_list_entry(d);
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -465,7 +466,7 @@ static int swap_process_new_swap(Manager *m, const char *device, int prio, bool
Zbigniew Jędrzejewski-Szmek e4a83a
                             st.st_rdev != udev_device_get_devnum(d))
Zbigniew Jędrzejewski-Szmek e4a83a
                                 continue;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                swap_add_one(m, p, device, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
+                swap_setup_unit(m, p, device, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
         }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         return r;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1091,15 +1092,17 @@ static int swap_load_proc_swaps(Manager *m, bool set_flags) {
Zbigniew Jędrzejewski-Szmek e4a83a
                         if (k == EOF)
Zbigniew Jędrzejewski-Szmek e4a83a
                                 break;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                        log_warning("Failed to parse /proc/swaps:%u", i);
Zbigniew Jędrzejewski-Szmek e4a83a
+                        log_warning("Failed to parse /proc/swaps:%u.", i);
Zbigniew Jędrzejewski-Szmek e4a83a
                         continue;
Zbigniew Jędrzejewski-Szmek e4a83a
                 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                 d = cunescape(dev);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (!d)
Zbigniew Jędrzejewski-Szmek e4a83a
-                        return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
+                        return log_oom();
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
+                device_found_node(m, d, true, DEVICE_FOUND_SWAP, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-                k = swap_process_new_swap(m, d, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
+                k = swap_process_new(m, d, prio, set_flags);
Zbigniew Jędrzejewski-Szmek e4a83a
                 if (k < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                         r = k;
Zbigniew Jędrzejewski-Szmek e4a83a
         }
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1151,6 +1154,9 @@ static int swap_dispatch_io(sd_event_source *source, int fd, uint32_t revents, v
Zbigniew Jędrzejewski-Szmek e4a83a
                                 break;
Zbigniew Jędrzejewski-Szmek e4a83a
                         }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
+                        if (swap->what)
Zbigniew Jędrzejewski-Szmek e4a83a
+                                device_found_node(m, swap->what, false, DEVICE_FOUND_SWAP, true);
Zbigniew Jędrzejewski-Szmek e4a83a
+
Zbigniew Jędrzejewski-Szmek e4a83a
                 } else if (swap->just_activated) {
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
                         /* New swap entry */
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1298,7 +1304,7 @@ fail:
Zbigniew Jędrzejewski-Szmek e4a83a
         return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-int swap_process_new_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
+int swap_process_device_new(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
         struct udev_list_entry *item = NULL, *first = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         _cleanup_free_ char *e = NULL;
Zbigniew Jędrzejewski-Szmek e4a83a
         const char *dn;
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -1341,7 +1347,7 @@ int swap_process_new_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
         return r;
Zbigniew Jędrzejewski-Szmek e4a83a
 }
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-int swap_process_removed_device(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
+int swap_process_device_remove(Manager *m, struct udev_device *dev) {
Zbigniew Jędrzejewski-Szmek e4a83a
         const char *dn;
Zbigniew Jędrzejewski-Szmek e4a83a
         int r = 0;
Zbigniew Jędrzejewski-Szmek e4a83a
         Swap *s;
Zbigniew Jędrzejewski-Szmek e4a83a
diff --git a/src/core/swap.h b/src/core/swap.h
Zbigniew Jędrzejewski-Szmek e4a83a
index 73e64d87a4..914a2dbccd 100644
Zbigniew Jędrzejewski-Szmek e4a83a
--- a/src/core/swap.h
Zbigniew Jędrzejewski-Szmek e4a83a
+++ b/src/core/swap.h
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -116,8 +116,8 @@ struct Swap {
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 extern const UnitVTable swap_vtable;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
-int swap_process_new_device(Manager *m, struct udev_device *dev);
Zbigniew Jędrzejewski-Szmek e4a83a
-int swap_process_removed_device(Manager *m, struct udev_device *dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+int swap_process_device_new(Manager *m, struct udev_device *dev);
Zbigniew Jędrzejewski-Szmek e4a83a
+int swap_process_device_remove(Manager *m, struct udev_device *dev);
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
 const char* swap_state_to_string(SwapState i) _const_;
Zbigniew Jędrzejewski-Szmek e4a83a
 SwapState swap_state_from_string(const char *s) _pure_;
Zbigniew Jędrzejewski-Szmek e4a83a
diff --git a/src/core/unit.c b/src/core/unit.c
Zbigniew Jędrzejewski-Szmek e4a83a
index 563f6fe850..a6558ee23b 100644
Zbigniew Jędrzejewski-Szmek e4a83a
--- a/src/core/unit.c
Zbigniew Jędrzejewski-Szmek e4a83a
+++ b/src/core/unit.c
Zbigniew Jędrzejewski-Szmek e4a83a
@@ -2843,7 +2843,6 @@ int unit_add_node_link(Unit *u, const char *what, bool wants) {
Zbigniew Jędrzejewski-Szmek e4a83a
                 return -ENOMEM;
Zbigniew Jędrzejewski-Szmek e4a83a
 
Zbigniew Jędrzejewski-Szmek e4a83a
         r = manager_load_unit(u->manager, e, NULL, NULL, &device);
Zbigniew Jędrzejewski-Szmek e4a83a
-
Zbigniew Jędrzejewski-Szmek e4a83a
         if (r < 0)
Zbigniew Jędrzejewski-Szmek e4a83a
                 return r;
Zbigniew Jędrzejewski-Szmek e4a83a