On 6/23/26 00:08, Nanzhe Zhao wrote:
> Large folio write path needs a subpage status bitmap and write
> pages pending counter, while keeping compatible with f2fs private
> flags.
>
> Move struct f2fs_folio_state to f2fs.h, add private_flags and
> subpage state bitmap, and change PAGE_PRIVATE functions to be
> compatible with f2fs_folio_state. Allocate f2fs_folio_state via kzalloc
> instead of kmem_cache, since the state size depends on the folio order.
>
> Note: Now if a path wants to use f2fs_folio_state, it must call
> `folio_has_ffs` instead of `folio_test_large`` to make check.
>
> Signed-off-by: Nanzhe Zhao <[email protected]>
> ---
> fs/f2fs/compress.c | 2 ++
> fs/f2fs/data.c | 46 ++++++++++++++++----------------
> fs/f2fs/f2fs.h | 66 ++++++++++++++++++++++++++++++++++++++--------
> 3 files changed, 80 insertions(+), 34 deletions(-)
>
> diff --git a/fs/f2fs/compress.c b/fs/f2fs/compress.c
> index 91855d91bbdd..de1305fcc6c1 100644
> --- a/fs/f2fs/compress.c
> +++ b/fs/f2fs/compress.c
> @@ -76,6 +76,8 @@ bool f2fs_is_compressed_page(struct folio *folio)
> {
> if (!folio->private)
> return false;
> + if (folio_has_ffs(folio))
> + return false;
> if (folio_test_f2fs_nonpointer(folio))
> return false;
private is a pointer for compressed page, so it will be better to
check folio_test_f2fs_nonpointer() first, and then folio_has_ffs().
if (folio_test_f2fs_nonpointer(folio))
return false;
if (folio_has_ffs(folio))
return false;
>
> diff --git a/fs/f2fs/data.c b/fs/f2fs/data.c
> index ac1cf4de3d62..23758c00758d 100644
> --- a/fs/f2fs/data.c
> +++ b/fs/f2fs/data.c
> @@ -32,20 +32,13 @@
>
> static struct kmem_cache *bio_post_read_ctx_cache;
> static struct kmem_cache *bio_entry_slab;
> -static struct kmem_cache *ffs_entry_slab;
> static mempool_t *bio_post_read_ctx_pool;
> static struct bio_set f2fs_bioset;
>
> -struct f2fs_folio_state {
> - spinlock_t state_lock;
> - unsigned int read_pages_pending;
> -};
> -
> struct f2fs_bio {
> struct work_struct work;
> struct bio bio;
> };
> -
> #define F2FS_BIO_POOL_SIZE NR_CURSEG_TYPE
>
> int __init f2fs_init_bioset(void)
> @@ -2514,15 +2507,30 @@ int f2fs_read_multi_pages(struct compress_ctx *cc,
> struct bio **bio_ret,
>
> static struct f2fs_folio_state *ffs_find_or_alloc(struct folio *folio)
> {
> - struct f2fs_folio_state *ffs = folio->private;
> + struct f2fs_folio_state *ffs;
> + unsigned int nr_subpages = folio_nr_pages(folio);
> + unsigned long private_flags = 0;
>
> - if (ffs)
> - return ffs;
> + f2fs_bug_on(F2FS_F_SB(folio), !folio_test_large(folio));
>
> - ffs = f2fs_kmem_cache_alloc(ffs_entry_slab,
> - GFP_NOIO | __GFP_ZERO, true, NULL);
> + if (folio_has_ffs(folio))
> + return (struct f2fs_folio_state *)folio->private;
> +
> + if (folio_test_private(folio) && folio_test_f2fs_nonpointer(folio))
Why this can happen? allocating large folio from other paths?
> + private_flags = (unsigned long)folio->private;
> +
> + ffs = kzalloc(struct_size(ffs, state, BITS_TO_LONGS(2 * nr_subpages)),
> + GFP_NOIO | __GFP_NOFAIL);
>
> spin_lock_init(&ffs->state_lock);
> + ffs->private_flags = private_flags;
> + if (folio_test_uptodate(folio))
> + bitmap_set(ffs->state, 0, nr_subpages);
> + if (folio_test_dirty(folio))
> + bitmap_set(ffs->state, nr_subpages, nr_subpages);
Can you describe ffs->state layout in its definition?
> +
> + if (folio_test_private(folio))
> + folio_detach_private(folio);
> folio_attach_private(folio, ffs);
> return ffs;
> }
> @@ -2531,7 +2539,7 @@ static void ffs_detach_free(struct folio *folio)
> {
> struct f2fs_folio_state *ffs;
>
> - if (!folio_test_large(folio)) {
> + if (!folio_has_ffs(folio)) {
> folio_detach_private(folio);
> return;
> }
> @@ -2541,7 +2549,8 @@ static void ffs_detach_free(struct folio *folio)
> return;
>
> WARN_ON_ONCE(ffs->read_pages_pending != 0);
> - kmem_cache_free(ffs_entry_slab, ffs);
> + WARN_ON_ONCE(atomic_read(&ffs->write_pages_pending));
> + kfree(ffs);
> }
>
> static int f2fs_read_data_large_folio(struct inode *inode,
> @@ -4571,21 +4580,12 @@ int __init f2fs_init_bio_entry_cache(void)
> if (!bio_entry_slab)
> return -ENOMEM;
>
> - ffs_entry_slab = f2fs_kmem_cache_create("f2fs_ffs_slab",
> - sizeof(struct f2fs_folio_state));
> -
> - if (!ffs_entry_slab) {
> - kmem_cache_destroy(bio_entry_slab);
> - return -ENOMEM;
> - }
> -
> return 0;
> }
>
> void f2fs_destroy_bio_entry_cache(void)
> {
> kmem_cache_destroy(bio_entry_slab);
> - kmem_cache_destroy(ffs_entry_slab);
> }
>
> static int f2fs_iomap_begin(struct inode *inode, loff_t offset, loff_t
> length,
> diff --git a/fs/f2fs/f2fs.h b/fs/f2fs/f2fs.h
> index f1774d4e18d2..e4778c17394e 100644
> --- a/fs/f2fs/f2fs.h
> +++ b/fs/f2fs/f2fs.h
> @@ -1620,6 +1620,15 @@ static inline void f2fs_clear_bit(unsigned int nr,
> char *addr);
> * Layout B: lowest bit should be 0
> * page.private is a wrapped pointer.
> */
> +
> +struct f2fs_folio_state {
> + spinlock_t state_lock;
> + unsigned int read_pages_pending;
> + atomic_t write_pages_pending;
> + unsigned long private_flags;
> + unsigned long state[];
> +};
> +
> enum {
> PAGE_PRIVATE_NOT_POINTER, /* private contains non-pointer
> data */
> PAGE_PRIVATE_ONGOING_MIGRATION, /* data page which is on-going
> migrating */
> @@ -1629,6 +1638,14 @@ enum {
> PAGE_PRIVATE_MAX
> };
>
> +static inline bool folio_has_ffs(const struct folio *folio)
> +{
> + unsigned long private = (unsigned long)folio->private;
> +
> + return folio_test_large(folio) && private &&
> + !(private & BIT(PAGE_PRIVATE_NOT_POINTER));
> +}
> +
> /* For compression */
> enum compress_algorithm_type {
> COMPRESS_LZO,
> @@ -2638,9 +2655,15 @@ static inline int inc_valid_block_count(struct
> f2fs_sb_info *sbi,
> #define PAGE_PRIVATE_GET_FUNC(name, flagname) \
> static inline bool folio_test_f2fs_##name(const struct folio *folio) \
Seems there are redundant codes below, let's have a try to wrap them w/ a macro
for cleanup?
> { \
> - unsigned long priv = (unsigned long)folio->private; \
> + unsigned long priv; \
> unsigned long v = (1UL << PAGE_PRIVATE_NOT_POINTER) | \
> (1UL << PAGE_PRIVATE_##flagname); \
> + if (folio_has_ffs(folio)) { \
> + struct f2fs_folio_state *ffs = folio->private; \
> + priv = ffs->private_flags; \
> + } else { \
> + priv = (unsigned long)folio->private; \
> + } \
> return (priv & v) == v; \
> } \
> static inline bool page_private_##name(struct page *page) \
> @@ -2655,7 +2678,10 @@ static inline void folio_set_f2fs_##name(struct folio
> *folio) \
> { \
> unsigned long v = (1UL << PAGE_PRIVATE_NOT_POINTER) | \
> (1UL << PAGE_PRIVATE_##flagname); \
> - if (!folio->private) \
> + if (folio_has_ffs(folio)) { \
> + struct f2fs_folio_state *ffs = folio->private; \
> + ffs->private_flags |= v; \
> + } else if (!folio->private) \
> folio_attach_private(folio, (void *)v); \
Ditto?
> else { \
> v |= (unsigned long)folio->private; \
> @@ -2673,13 +2699,18 @@ static inline void set_page_private_##name(struct
> page *page) \
> #define PAGE_PRIVATE_CLEAR_FUNC(name, flagname) \
> static inline void folio_clear_f2fs_##name(struct folio *folio)
> \
> { \
> - unsigned long v = (unsigned long)folio->private; \
> + if (folio_has_ffs(folio)) { \
> + struct f2fs_folio_state *ffs = folio->private; \
> + ffs->private_flags &= ~(1UL << PAGE_PRIVATE_##flagname); \
> + } else { \
> + unsigned long v = (unsigned long)folio->private; \
> \
> - v &= ~(1UL << PAGE_PRIVATE_##flagname); \
> - if (v == (1UL << PAGE_PRIVATE_NOT_POINTER)) \
> - folio_detach_private(folio); \
> - else \
> - folio->private = (void *)v; \
> + v &= ~(1UL << PAGE_PRIVATE_##flagname); \
> + if (v == (1UL << PAGE_PRIVATE_NOT_POINTER)) \
> + folio_detach_private(folio); \
> + else \
> + folio->private = (void *)v; \
> + } \
> } \
> static inline void clear_page_private_##name(struct page *page) \
> { \
> @@ -2705,7 +2736,15 @@ PAGE_PRIVATE_CLEAR_FUNC(atomic, ATOMIC_WRITE);
>
> static inline unsigned long folio_get_f2fs_data(struct folio *folio)
> {
> - unsigned long data = (unsigned long)folio->private;
> + unsigned long data;
> +
> + if (folio_has_ffs(folio)) {
> + struct f2fs_folio_state *ffs = folio->private;
> +
> + data = ffs->private_flags;
> + } else {
> + data = (unsigned long)folio->private;
> + }
Ditto?
>
> if (!test_bit(PAGE_PRIVATE_NOT_POINTER, &data))
> return 0;
> @@ -2716,10 +2755,15 @@ static inline void folio_set_f2fs_data(struct folio
> *folio, unsigned long data)
> {
> data = (1UL << PAGE_PRIVATE_NOT_POINTER) | (data << PAGE_PRIVATE_MAX);
>
> - if (!folio_test_private(folio))
> + if (folio_has_ffs(folio)) {
> + struct f2fs_folio_state *ffs = folio->private;
> +
> + ffs->private_flags |= data;
> + } else if (!folio_test_private(folio)) {
> folio_attach_private(folio, (void *)data);
Ditto?
Thanks,
> - else
> + } else {
> folio->private = (void *)((unsigned long)folio->private | data);
> + }
> }
>
> static inline void dec_valid_block_count(struct f2fs_sb_info *sbi,
_______________________________________________
Linux-f2fs-devel mailing list
[email protected]
https://lists.sourceforge.net/lists/listinfo/linux-f2fs-devel