]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
ext4: limit the maximum folio order
authorZhang Yi <yi.zhang@huawei.com>
Mon, 7 Jul 2025 14:08:14 +0000 (22:08 +0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 20 Aug 2025 16:41:23 +0000 (18:41 +0200)
[ Upstream commit b12f423d598fd874df9ecfb2436789d582fda8e6 ]

In environments with a page size of 64KB, the maximum size of a folio
can reach up to 128MB. Consequently, during the write-back of folios,
the 'rsv_blocks' will be overestimated to 1,577, which can make
pressure on the journal space where the journal is small. This can
easily exceed the limit of a single transaction. Besides, an excessively
large folio is meaningless and will instead increase the overhead of
traversing the bhs within the folio. Therefore, limit the maximum order
of a folio to 2048 filesystem blocks.

Reported-by: Naresh Kamboju <naresh.kamboju@linaro.org>
Reported-by: Joseph Qi <jiangqi903@gmail.com>
Closes: https://lore.kernel.org/linux-ext4/CA+G9fYsyYQ3ZL4xaSg1-Tt5Evto7Zd+hgNWZEa9cQLbahA1+xg@mail.gmail.com/
Signed-off-by: Zhang Yi <yi.zhang@huawei.com>
Tested-by: Joseph Qi <joseph.qi@linux.alibaba.com>
Reviewed-by: Jan Kara <jack@suse.cz>
Link: https://patch.msgid.link/20250707140814.542883-12-yi.zhang@huaweicloud.com
Signed-off-by: Theodore Ts'o <tytso@mit.edu>
Signed-off-by: Sasha Levin <sashal@kernel.org>
fs/ext4/ext4.h
fs/ext4/ialloc.c
fs/ext4/inode.c

index 18373de980f27aad43fae87b0b435cdea6c3c9ba..fe3366e98493b8f3fd3b93a1f5e8f7dccc4573c1 100644 (file)
@@ -3020,7 +3020,7 @@ int ext4_walk_page_buffers(handle_t *handle,
                                     struct buffer_head *bh));
 int do_journal_get_write_access(handle_t *handle, struct inode *inode,
                                struct buffer_head *bh);
-bool ext4_should_enable_large_folio(struct inode *inode);
+void ext4_set_inode_mapping_order(struct inode *inode);
 #define FALL_BACK_TO_NONDELALLOC 1
 #define CONVERT_INLINE_DATA     2
 
index 79aa3df8d0197b227cb9d5af401556bd15bed3d0..df4051613b290a78728176d3b8198ad90de84977 100644 (file)
@@ -1335,8 +1335,7 @@ got:
                }
        }
 
-       if (ext4_should_enable_large_folio(inode))
-               mapping_set_large_folios(inode->i_mapping);
+       ext4_set_inode_mapping_order(inode);
 
        ext4_update_inode_fsync_trans(handle, inode, 1);
 
index ee4129b5ecce33997d39235c2abc973fe1a6fde9..0f316632b8dd65a347885a558536d268c36833d2 100644 (file)
@@ -5107,7 +5107,7 @@ error:
        return -EFSCORRUPTED;
 }
 
-bool ext4_should_enable_large_folio(struct inode *inode)
+static bool ext4_should_enable_large_folio(struct inode *inode)
 {
        struct super_block *sb = inode->i_sb;
 
@@ -5124,6 +5124,22 @@ bool ext4_should_enable_large_folio(struct inode *inode)
        return true;
 }
 
+/*
+ * Limit the maximum folio order to 2048 blocks to prevent overestimation
+ * of reserve handle credits during the folio writeback in environments
+ * where the PAGE_SIZE exceeds 4KB.
+ */
+#define EXT4_MAX_PAGECACHE_ORDER(i)            \
+               umin(MAX_PAGECACHE_ORDER, (11 + (i)->i_blkbits - PAGE_SHIFT))
+void ext4_set_inode_mapping_order(struct inode *inode)
+{
+       if (!ext4_should_enable_large_folio(inode))
+               return;
+
+       mapping_set_folio_order_range(inode->i_mapping, 0,
+                                     EXT4_MAX_PAGECACHE_ORDER(inode));
+}
+
 struct inode *__ext4_iget(struct super_block *sb, unsigned long ino,
                          ext4_iget_flags flags, const char *function,
                          unsigned int line)
@@ -5441,8 +5457,8 @@ struct inode *__ext4_iget(struct super_block *sb, unsigned long ino,
                ret = -EFSCORRUPTED;
                goto bad_inode;
        }
-       if (ext4_should_enable_large_folio(inode))
-               mapping_set_large_folios(inode->i_mapping);
+
+       ext4_set_inode_mapping_order(inode);
 
        ret = check_igot_inode(inode, flags, function, line);
        /*