]> git.ipfire.org Git - thirdparty/systemd.git/commitdiff
dissect: tighten block device checks a bit
authorLennart Poettering <lennart@poettering.net>
Thu, 5 Apr 2018 11:03:37 +0000 (13:03 +0200)
committerLennart Poettering <lennart@poettering.net>
Thu, 5 Apr 2018 11:03:37 +0000 (13:03 +0200)
This extends on #8609, and makes two changes:

1. We'll now explicitly check that the child devices of a block device
   we are interested in (i.e. the partitions) are block devices themselves.
   On newer kernels the mmc rpmb stuff is actually exposed as char rather
   than block device as before, and they probably should have been that in
   the first place. By adding this check we'll hence filter out these weird
   devices through a second rule too, that hopefully makes things a bit
   more future-proof, should more devices like this be added eventually,
   or other subsystems do a similar thing.

2. When counting partitions we'll now also check the devnum of the
   device being non-null, which we already do when matching up the devices
   in the second iteration. This should make things more robust, and
   prevent other kinds of miscounting, which after all was the main
   issue #8609 fixed.

src/shared/dissect-image.c

index e2baa4497c0a5ca9f5db3f3fac1fbce33e705d57..c29d70fa042c501e1e429a8ffe6b44414c2f1131 100644 (file)
@@ -111,11 +111,23 @@ not_found:
 /* Detect RPMB and Boot partitions, which are not listed by blkid.
  * See https://github.com/systemd/systemd/issues/5806. */
 static bool device_is_mmc_special_partition(struct udev_device *d) {
-        const char *sysname = udev_device_get_sysname(d);
+        const char *sysname;
+
+        sysname = udev_device_get_sysname(d);
         return (sysname && startswith(sysname, "mmcblk") &&
                 (endswith(sysname, "rpmb") || endswith(sysname, "boot0") || endswith(sysname, "boot1")));
 }
 
+static bool device_is_block(struct udev_device *d) {
+        const char *ss;
+
+        ss = udev_device_get_subsystem(d);
+        if (!ss)
+                return false;
+
+        return streq(ss, "block");
+}
+
 int dissect_image(
                 int fd,
                 const void *root_hash,
@@ -289,11 +301,19 @@ int dissect_image(
                 first = udev_enumerate_get_list_entry(e);
                 udev_list_entry_foreach(item, first) {
                         _cleanup_udev_device_unref_ struct udev_device *q;
+                        dev_t qn;
 
                         q = udev_device_new_from_syspath(udev, udev_list_entry_get_name(item));
                         if (!q)
                                 return -errno;
 
+                        qn = udev_device_get_devnum(q);
+                        if (major(qn) == 0)
+                                continue;
+
+                        if (!device_is_block(q))
+                                continue;
+
                         if (device_is_mmc_special_partition(q))
                                 continue;
                         n++;
@@ -371,6 +391,9 @@ int dissect_image(
                 if (st.st_rdev == qn)
                         continue;
 
+                if (!device_is_block(q))
+                        continue;
+
                 if (device_is_mmc_special_partition(q))
                         continue;