]> www.infradead.org Git - users/dwmw2/linux.git/commitdiff
nilfs2: fix inode number range checks
authorRyusuke Konishi <konishi.ryusuke@gmail.com>
Sun, 23 Jun 2024 05:11:33 +0000 (14:11 +0900)
committerAndrew Morton <akpm@linux-foundation.org>
Wed, 3 Jul 2024 19:29:24 +0000 (12:29 -0700)
Patch series "nilfs2: fix potential issues related to reserved inodes".

This series fixes one use-after-free issue reported by syzbot, caused by
nilfs2's internal inode being exposed in the namespace on a corrupted
filesystem, and a couple of flaws that cause problems if the starting
number of non-reserved inodes written in the on-disk super block is
intentionally (or corruptly) changed from its default value.

This patch (of 3):

In the current implementation of nilfs2, "nilfs->ns_first_ino", which
gives the first non-reserved inode number, is read from the superblock,
but its lower limit is not checked.

As a result, if a number that overlaps with the inode number range of
reserved inodes such as the root directory or metadata files is set in the
super block parameter, the inode number test macros (NILFS_MDT_INODE and
NILFS_VALID_INODE) will not function properly.

In addition, these test macros use left bit-shift calculations using with
the inode number as the shift count via the BIT macro, but the result of a
shift calculation that exceeds the bit width of an integer is undefined in
the C specification, so if "ns_first_ino" is set to a large value other
than the default value NILFS_USER_INO (=11), the macros may potentially
malfunction depending on the environment.

Fix these issues by checking the lower bound of "nilfs->ns_first_ino" and
by preventing bit shifts equal to or greater than the NILFS_USER_INO
constant in the inode number test macros.

Also, change the type of "ns_first_ino" from signed integer to unsigned
integer to avoid the need for type casting in comparisons such as the
lower bound check introduced this time.

Link: https://lkml.kernel.org/r/20240623051135.4180-1-konishi.ryusuke@gmail.com
Link: https://lkml.kernel.org/r/20240623051135.4180-2-konishi.ryusuke@gmail.com
Signed-off-by: Ryusuke Konishi <konishi.ryusuke@gmail.com>
Cc: Hillf Danton <hdanton@sina.com>
Cc: Jan Kara <jack@suse.cz>
Cc: Matthew Wilcox (Oracle) <willy@infradead.org>
Cc: <stable@vger.kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
fs/nilfs2/nilfs.h
fs/nilfs2/the_nilfs.c
fs/nilfs2/the_nilfs.h

index 728e90be3570b7f9ce71b5d08d2e784b4a7dd537..7e39e277c77f00e3f61035caec275dfe988492ca 100644 (file)
@@ -116,9 +116,10 @@ enum {
 #define NILFS_FIRST_INO(sb) (((struct the_nilfs *)sb->s_fs_info)->ns_first_ino)
 
 #define NILFS_MDT_INODE(sb, ino) \
-       ((ino) < NILFS_FIRST_INO(sb) && (NILFS_MDT_INO_BITS & BIT(ino)))
+       ((ino) < NILFS_USER_INO && (NILFS_MDT_INO_BITS & BIT(ino)))
 #define NILFS_VALID_INODE(sb, ino) \
-       ((ino) >= NILFS_FIRST_INO(sb) || (NILFS_SYS_INO_BITS & BIT(ino)))
+       ((ino) >= NILFS_FIRST_INO(sb) ||                                \
+        ((ino) < NILFS_USER_INO && (NILFS_SYS_INO_BITS & BIT(ino))))
 
 /**
  * struct nilfs_transaction_info: context information for synchronization
index f41d7b6d432c6bf5f845229ba5295168588286c8..e44dde57ab652bb02447f1e79490a630a36b900f 100644 (file)
@@ -452,6 +452,12 @@ static int nilfs_store_disk_layout(struct the_nilfs *nilfs,
        }
 
        nilfs->ns_first_ino = le32_to_cpu(sbp->s_first_ino);
+       if (nilfs->ns_first_ino < NILFS_USER_INO) {
+               nilfs_err(nilfs->ns_sb,
+                         "too small lower limit for non-reserved inode numbers: %u",
+                         nilfs->ns_first_ino);
+               return -EINVAL;
+       }
 
        nilfs->ns_blocks_per_segment = le32_to_cpu(sbp->s_blocks_per_segment);
        if (nilfs->ns_blocks_per_segment < NILFS_SEG_MIN_BLOCKS) {
index 85da0629415dfe0cba663da634c3d40e04877f70..1e829ed7b0ef56ffe57d099dfa3a18f674f46f55 100644 (file)
@@ -182,7 +182,7 @@ struct the_nilfs {
        unsigned long           ns_nrsvsegs;
        unsigned long           ns_first_data_block;
        int                     ns_inode_size;
-       int                     ns_first_ino;
+       unsigned int            ns_first_ino;
        u32                     ns_crc_seed;
 
        /* /sys/fs/<nilfs>/<device> */