grub-devel
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

[PATCH] disk/mdraid1x_linux: Prevent infinite recursion


From: Lidong Chen
Subject: [PATCH] disk/mdraid1x_linux: Prevent infinite recursion
Date: Thu, 29 Feb 2024 22:13:50 +0000

The test corpus for version-1 RAID generated an infinite recursion
in grub_partition_iterate() while attempting to read the superblock.
The reason for the issue was that the data region overlapped with
the superblock.

The infinite call loop looks like this:
grub_partition_iterate() -> partmap->iterate() ->
  -> grub_disk_read() -> grub_disk_read_small() ->
  -> grub_disk_read_small_real() -> grub_diskfilter_read() ->
  -> read_lv() -> read_segment() -> grub_diskfilter_read_node() ->
  -> grub_disk_read() -> grub_disk_read_small() ->...

The fix adds checks for both the superblock region and the data
region when parsing the superblock metadata in grub_mdraid_detect().

Signed-off-by: Lidong Chen <lidong.chen@oracle.com>
---
 grub-core/disk/mdraid1x_linux.c | 66 +++++++++++++++++++++++++++++++++
 1 file changed, 66 insertions(+)

diff --git a/grub-core/disk/mdraid1x_linux.c b/grub-core/disk/mdraid1x_linux.c
index 72e5cb6f4..c09efd5b9 100644
--- a/grub-core/disk/mdraid1x_linux.c
+++ b/grub-core/disk/mdraid1x_linux.c
@@ -103,6 +103,9 @@ struct grub_raid_super_1x
 
 #define WriteMostly1    1      /* Mask for writemostly flag in above devflags. 
 */
 
+#define GRUB_MD_SECTOR_SHIFT   9       /* Follow Linux kernel v6.8*/
+#define GRUB_MD_SECTOR_SIZE    (1 << GRUB_MD_SECTOR_SHIFT)
+
 static struct grub_diskfilter_vg *
 grub_mdraid_detect (grub_disk_t disk,
                    struct grub_diskfilter_pv_id *id,
@@ -129,6 +132,7 @@ grub_mdraid_detect (grub_disk_t disk,
       grub_uint32_t level;
       struct grub_diskfilter_vg *array;
       char *uuid;
+      grub_uint64_t sb_sz, data_end, sb_end;
 
       if (size == GRUB_DISK_SIZE_UNKNOWN && minor_version == 0)
        continue;
@@ -154,6 +158,68 @@ grub_mdraid_detect (grub_disk_t disk,
          || grub_le_to_cpu64 (sb.super_offset) != sector)
        continue;
 
+      /* Follows the Linux kernel's data_size validation from v6.8-rc5. */
+      if (sb.data_size < 10)
+       {
+         grub_dprintf ("mdraid1x", "The data_size is too small\n");
+         return NULL;
+       }
+
+      /*
+       * Total size of superblock: 256 bytes plus 2 bytes per device
+       * in the array.
+       */
+      sb_sz = sizeof (struct grub_raid_super_1x) +
+             grub_le_to_cpu64 (sb.raid_disks) * 2;
+
+      sb_end = grub_le_to_cpu64 (sb.super_offset) +
+              (ALIGN_UP(sb_sz, GRUB_MD_SECTOR_SIZE) >> GRUB_MD_SECTOR_SHIFT);
+
+      data_end = grub_le_to_cpu64 (sb.data_offset) +
+                grub_le_to_cpu64 (sb.data_size);
+
+      /* In minor versions 1 and 2, superblock is positioned before data. */
+      if (minor_version)
+       {
+         if (grub_le_to_cpu64 (sb.data_offset) < sb_end)
+           {
+             grub_dprintf ("mdraid1x",
+                           "The superblock either overlaps with the data "
+                           "or is behind it.\n");
+             return NULL;
+           }
+
+         if (data_end > size)
+           {
+             grub_dprintf ("mdraid1x",
+                           "The data region ends at %" PRIuGRUB_UINT64_T ", "
+                           "past the end of the disk (%" PRIuGRUB_UINT64_T 
")\n",
+                           data_end, size);
+             return NULL;
+           }
+       }
+      else
+       {
+         /* In minor version 0, superblock is at the end of the device. */
+         if (grub_le_to_cpu64 (sb.super_offset) < data_end)
+           {
+             grub_dprintf ("mdraid1x",
+                           "The data either overlaps with the superblock "
+                           "or is behind it.\n");
+             return NULL;
+           }
+
+         if (sb_end > size)
+           {
+             grub_dprintf ("mdraid1x",
+                           "The superblock region ends at "
+                           "%" PRIuGRUB_UINT64_T ", past the end of "
+                           "the disk (%" PRIuGRUB_UINT64_T ")\n",
+                           sb_end, size);
+             return NULL;
+           }
+       }
+
       if (sb.major_version != grub_cpu_to_le32_compile_time (1))
        /* Unsupported version.  */
        return NULL;
-- 
2.34.1




reply via email to

[Prev in Thread] Current Thread [Next in Thread]