ryantimwilson / rpms / systemd

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