md/raid5: Convert stripe_head's "dev" to flexible array member
authorKees Cook <keescook@chromium.org>
Fri, 12 May 2023 21:22:12 +0000 (14:22 -0700)
committerKees Cook <keescook@chromium.org>
Tue, 30 May 2023 23:41:52 +0000 (16:41 -0700)
Replace old-style 1-element array of "dev" in struct stripe_head with
modern C99 flexible array. In the future, we can additionally annotate
it with the run-time size, found in the "disks" member.

Cc: Song Liu <song@kernel.org>
Cc: linux-raid@vger.kernel.org
Reviewed-by: Christoph Hellwig <hch@lst.de>
Acked-by: Song Liu <song@kernel.org>
Signed-off-by: Kees Cook <keescook@chromium.org>
Link: https://lore.kernel.org/lkml/20230522212114.gonna.589-kees@kernel.org/
---
It looks like this memory calculation:

        memory = conf->min_nr_stripes * (sizeof(struct stripe_head) +
                 max_disks * ((sizeof(struct bio) + PAGE_SIZE))) / 1024;

... was already buggy (i.e. it included the single "dev" bytes in the
result). However, I'm not entirely sure if that is the right analysis,
since "dev" is not related to struct bio nor PAGE_SIZE?

drivers/md/raid5.c
drivers/md/raid5.h

index 4739ed891e756eb3a8af3e7c48c6be0950b82db7..64865f9dd3f5e0b24ee240e917c2038a428754ef 100644 (file)
@@ -2433,7 +2433,7 @@ static int grow_stripes(struct r5conf *conf, int num)
 
        conf->active_name = 0;
        sc = kmem_cache_create(conf->cache_name[conf->active_name],
-                              sizeof(struct stripe_head)+(devs-1)*sizeof(struct r5dev),
+                              struct_size_t(struct stripe_head, dev, devs),
                               0, 0, NULL);
        if (!sc)
                return 1;
@@ -2559,7 +2559,7 @@ static int resize_stripes(struct r5conf *conf, int newsize)
 
        /* Step 1 */
        sc = kmem_cache_create(conf->cache_name[1-conf->active_name],
-                              sizeof(struct stripe_head)+(newsize-1)*sizeof(struct r5dev),
+                              struct_size_t(struct stripe_head, dev, newsize),
                               0, 0, NULL);
        if (!sc)
                return -ENOMEM;
index e873938a6125d3a834df5e95bc12bd3940ee5e03..6a92fafb0748ed96ecfe55b174dca95776e4f3a6 100644 (file)
@@ -268,7 +268,7 @@ struct stripe_head {
                unsigned long   flags;
                u32             log_checksum;
                unsigned short  write_hint;
-       } dev[1]; /* allocated with extra space depending of RAID geometry */
+       } dev[]; /* allocated depending of RAID geometry ("disks" member) */
 };
 
 /* stripe_head_state - collects and tracks the dynamic state of a stripe_head