Skip to content

Commit d151e8b

Browse files
committed
Merge tag 'iomap-6.3-merge-1' of git://git.kernel.org/pub/scm/fs/xfs/xfs-linux
Pull iomap updates from Darrick Wong: "This is mostly rearranging things to make life easier for gfs2, nothing all that mindblowing for this release. - Change when the iomap page_done function is called so that we still have a locked folio in the success case. This fixes a writeback race in gfs2 - Change when the iomap page_prepare function is called so that gfs2 can recover from OOM scenarios more gracefully - Rename the iomap page_ops to folio_ops, since they operate on folios now" * tag 'iomap-6.3-merge-1' of git://git.kernel.org/pub/scm/fs/xfs/xfs-linux: iomap: Rename page_ops to folio_ops iomap: Rename page_prepare handler to get_folio iomap: Add __iomap_get_folio helper iomap/gfs2: Get page in page_prepare handler iomap: Add iomap_get_folio helper iomap: Rename page_done handler to put_folio iomap/gfs2: Unlock and put folio in page_done handler iomap: Add __iomap_put_folio helper
2 parents 8762069 + 471859f commit d151e8b

File tree

4 files changed

+103
-57
lines changed

4 files changed

+103
-57
lines changed

fs/gfs2/bmap.c

Lines changed: 26 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -956,36 +956,50 @@ static int __gfs2_iomap_get(struct inode *inode, loff_t pos, loff_t length,
956956
goto out;
957957
}
958958

959-
static int gfs2_iomap_page_prepare(struct inode *inode, loff_t pos,
960-
unsigned len)
959+
static struct folio *
960+
gfs2_iomap_get_folio(struct iomap_iter *iter, loff_t pos, unsigned len)
961961
{
962+
struct inode *inode = iter->inode;
962963
unsigned int blockmask = i_blocksize(inode) - 1;
963964
struct gfs2_sbd *sdp = GFS2_SB(inode);
964965
unsigned int blocks;
966+
struct folio *folio;
967+
int status;
965968

966969
blocks = ((pos & blockmask) + len + blockmask) >> inode->i_blkbits;
967-
return gfs2_trans_begin(sdp, RES_DINODE + blocks, 0);
970+
status = gfs2_trans_begin(sdp, RES_DINODE + blocks, 0);
971+
if (status)
972+
return ERR_PTR(status);
973+
974+
folio = iomap_get_folio(iter, pos);
975+
if (IS_ERR(folio))
976+
gfs2_trans_end(sdp);
977+
return folio;
968978
}
969979

970-
static void gfs2_iomap_page_done(struct inode *inode, loff_t pos,
971-
unsigned copied, struct page *page)
980+
static void gfs2_iomap_put_folio(struct inode *inode, loff_t pos,
981+
unsigned copied, struct folio *folio)
972982
{
973983
struct gfs2_trans *tr = current->journal_info;
974984
struct gfs2_inode *ip = GFS2_I(inode);
975985
struct gfs2_sbd *sdp = GFS2_SB(inode);
976986

977-
if (page && !gfs2_is_stuffed(ip))
978-
gfs2_page_add_databufs(ip, page, offset_in_page(pos), copied);
987+
if (!gfs2_is_stuffed(ip))
988+
gfs2_page_add_databufs(ip, &folio->page, offset_in_page(pos),
989+
copied);
990+
991+
folio_unlock(folio);
992+
folio_put(folio);
979993

980994
if (tr->tr_num_buf_new)
981995
__mark_inode_dirty(inode, I_DIRTY_DATASYNC);
982996

983997
gfs2_trans_end(sdp);
984998
}
985999

986-
static const struct iomap_page_ops gfs2_iomap_page_ops = {
987-
.page_prepare = gfs2_iomap_page_prepare,
988-
.page_done = gfs2_iomap_page_done,
1000+
static const struct iomap_folio_ops gfs2_iomap_folio_ops = {
1001+
.get_folio = gfs2_iomap_get_folio,
1002+
.put_folio = gfs2_iomap_put_folio,
9891003
};
9901004

9911005
static int gfs2_iomap_begin_write(struct inode *inode, loff_t pos,
@@ -1061,7 +1075,7 @@ static int gfs2_iomap_begin_write(struct inode *inode, loff_t pos,
10611075
}
10621076

10631077
if (gfs2_is_stuffed(ip) || gfs2_is_jdata(ip))
1064-
iomap->page_ops = &gfs2_iomap_page_ops;
1078+
iomap->folio_ops = &gfs2_iomap_folio_ops;
10651079
return 0;
10661080

10671081
out_trans_end:
@@ -1277,7 +1291,7 @@ int gfs2_alloc_extent(struct inode *inode, u64 lblock, u64 *dblock,
12771291
/*
12781292
* NOTE: Never call gfs2_block_zero_range with an open transaction because it
12791293
* uses iomap write to perform its actions, which begin their own transactions
1280-
* (iomap_begin, page_prepare, etc.)
1294+
* (iomap_begin, get_folio, etc.)
12811295
*/
12821296
static int gfs2_block_zero_range(struct inode *inode, loff_t from,
12831297
unsigned int length)

fs/iomap/buffered-io.c

Lines changed: 60 additions & 31 deletions
Original file line numberDiff line numberDiff line change
@@ -457,6 +457,33 @@ bool iomap_is_partially_uptodate(struct folio *folio, size_t from, size_t count)
457457
}
458458
EXPORT_SYMBOL_GPL(iomap_is_partially_uptodate);
459459

460+
/**
461+
* iomap_get_folio - get a folio reference for writing
462+
* @iter: iteration structure
463+
* @pos: start offset of write
464+
*
465+
* Returns a locked reference to the folio at @pos, or an error pointer if the
466+
* folio could not be obtained.
467+
*/
468+
struct folio *iomap_get_folio(struct iomap_iter *iter, loff_t pos)
469+
{
470+
unsigned fgp = FGP_LOCK | FGP_WRITE | FGP_CREAT | FGP_STABLE | FGP_NOFS;
471+
struct folio *folio;
472+
473+
if (iter->flags & IOMAP_NOWAIT)
474+
fgp |= FGP_NOWAIT;
475+
476+
folio = __filemap_get_folio(iter->inode->i_mapping, pos >> PAGE_SHIFT,
477+
fgp, mapping_gfp_mask(iter->inode->i_mapping));
478+
if (folio)
479+
return folio;
480+
481+
if (iter->flags & IOMAP_NOWAIT)
482+
return ERR_PTR(-EAGAIN);
483+
return ERR_PTR(-ENOMEM);
484+
}
485+
EXPORT_SYMBOL_GPL(iomap_get_folio);
486+
460487
bool iomap_release_folio(struct folio *folio, gfp_t gfp_flags)
461488
{
462489
trace_iomap_release_folio(folio->mapping->host, folio_pos(folio),
@@ -575,6 +602,30 @@ static int __iomap_write_begin(const struct iomap_iter *iter, loff_t pos,
575602
return 0;
576603
}
577604

605+
static struct folio *__iomap_get_folio(struct iomap_iter *iter, loff_t pos,
606+
size_t len)
607+
{
608+
const struct iomap_folio_ops *folio_ops = iter->iomap.folio_ops;
609+
610+
if (folio_ops && folio_ops->get_folio)
611+
return folio_ops->get_folio(iter, pos, len);
612+
else
613+
return iomap_get_folio(iter, pos);
614+
}
615+
616+
static void __iomap_put_folio(struct iomap_iter *iter, loff_t pos, size_t ret,
617+
struct folio *folio)
618+
{
619+
const struct iomap_folio_ops *folio_ops = iter->iomap.folio_ops;
620+
621+
if (folio_ops && folio_ops->put_folio) {
622+
folio_ops->put_folio(iter->inode, pos, ret, folio);
623+
} else {
624+
folio_unlock(folio);
625+
folio_put(folio);
626+
}
627+
}
628+
578629
static int iomap_write_begin_inline(const struct iomap_iter *iter,
579630
struct folio *folio)
580631
{
@@ -587,15 +638,11 @@ static int iomap_write_begin_inline(const struct iomap_iter *iter,
587638
static int iomap_write_begin(struct iomap_iter *iter, loff_t pos,
588639
size_t len, struct folio **foliop)
589640
{
590-
const struct iomap_page_ops *page_ops = iter->iomap.page_ops;
641+
const struct iomap_folio_ops *folio_ops = iter->iomap.folio_ops;
591642
const struct iomap *srcmap = iomap_iter_srcmap(iter);
592643
struct folio *folio;
593-
unsigned fgp = FGP_LOCK | FGP_WRITE | FGP_CREAT | FGP_STABLE | FGP_NOFS;
594644
int status = 0;
595645

596-
if (iter->flags & IOMAP_NOWAIT)
597-
fgp |= FGP_NOWAIT;
598-
599646
BUG_ON(pos + len > iter->iomap.offset + iter->iomap.length);
600647
if (srcmap != &iter->iomap)
601648
BUG_ON(pos + len > srcmap->offset + srcmap->length);
@@ -606,18 +653,9 @@ static int iomap_write_begin(struct iomap_iter *iter, loff_t pos,
606653
if (!mapping_large_folio_support(iter->inode->i_mapping))
607654
len = min_t(size_t, len, PAGE_SIZE - offset_in_page(pos));
608655

609-
if (page_ops && page_ops->page_prepare) {
610-
status = page_ops->page_prepare(iter->inode, pos, len);
611-
if (status)
612-
return status;
613-
}
614-
615-
folio = __filemap_get_folio(iter->inode->i_mapping, pos >> PAGE_SHIFT,
616-
fgp, mapping_gfp_mask(iter->inode->i_mapping));
617-
if (!folio) {
618-
status = (iter->flags & IOMAP_NOWAIT) ? -EAGAIN : -ENOMEM;
619-
goto out_no_page;
620-
}
656+
folio = __iomap_get_folio(iter, pos, len);
657+
if (IS_ERR(folio))
658+
return PTR_ERR(folio);
621659

622660
/*
623661
* Now we have a locked folio, before we do anything with it we need to
@@ -629,9 +667,9 @@ static int iomap_write_begin(struct iomap_iter *iter, loff_t pos,
629667
* could do the wrong thing here (zero a page range incorrectly or fail
630668
* to zero) and corrupt data.
631669
*/
632-
if (page_ops && page_ops->iomap_valid) {
633-
bool iomap_valid = page_ops->iomap_valid(iter->inode,
634-
&iter->iomap);
670+
if (folio_ops && folio_ops->iomap_valid) {
671+
bool iomap_valid = folio_ops->iomap_valid(iter->inode,
672+
&iter->iomap);
635673
if (!iomap_valid) {
636674
iter->iomap.flags |= IOMAP_F_STALE;
637675
status = 0;
@@ -656,13 +694,9 @@ static int iomap_write_begin(struct iomap_iter *iter, loff_t pos,
656694
return 0;
657695

658696
out_unlock:
659-
folio_unlock(folio);
660-
folio_put(folio);
697+
__iomap_put_folio(iter, pos, 0, folio);
661698
iomap_write_failed(iter->inode, pos, len);
662699

663-
out_no_page:
664-
if (page_ops && page_ops->page_done)
665-
page_ops->page_done(iter->inode, pos, 0, NULL);
666700
return status;
667701
}
668702

@@ -712,7 +746,6 @@ static size_t iomap_write_end_inline(const struct iomap_iter *iter,
712746
static size_t iomap_write_end(struct iomap_iter *iter, loff_t pos, size_t len,
713747
size_t copied, struct folio *folio)
714748
{
715-
const struct iomap_page_ops *page_ops = iter->iomap.page_ops;
716749
const struct iomap *srcmap = iomap_iter_srcmap(iter);
717750
loff_t old_size = iter->inode->i_size;
718751
size_t ret;
@@ -735,14 +768,10 @@ static size_t iomap_write_end(struct iomap_iter *iter, loff_t pos, size_t len,
735768
i_size_write(iter->inode, pos + ret);
736769
iter->iomap.flags |= IOMAP_F_SIZE_CHANGED;
737770
}
738-
folio_unlock(folio);
771+
__iomap_put_folio(iter, pos, ret, folio);
739772

740773
if (old_size < pos)
741774
pagecache_isize_extended(iter->inode, old_size, pos);
742-
if (page_ops && page_ops->page_done)
743-
page_ops->page_done(iter->inode, pos, ret, &folio->page);
744-
folio_put(folio);
745-
746775
if (ret < len)
747776
iomap_write_failed(iter->inode, pos + ret, len - ret);
748777
return ret;

fs/xfs/xfs_iomap.c

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -83,7 +83,7 @@ xfs_iomap_valid(
8383
return true;
8484
}
8585

86-
static const struct iomap_page_ops xfs_iomap_page_ops = {
86+
static const struct iomap_folio_ops xfs_iomap_folio_ops = {
8787
.iomap_valid = xfs_iomap_valid,
8888
};
8989

@@ -133,7 +133,7 @@ xfs_bmbt_to_iomap(
133133
iomap->flags |= IOMAP_F_DIRTY;
134134

135135
iomap->validity_cookie = sequence_cookie;
136-
iomap->page_ops = &xfs_iomap_page_ops;
136+
iomap->folio_ops = &xfs_iomap_folio_ops;
137137
return 0;
138138
}
139139

include/linux/iomap.h

Lines changed: 15 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
struct address_space;
1414
struct fiemap_extent_info;
1515
struct inode;
16+
struct iomap_iter;
1617
struct iomap_dio;
1718
struct iomap_writepage_ctx;
1819
struct iov_iter;
@@ -84,7 +85,7 @@ struct vm_fault;
8485
*/
8586
#define IOMAP_NULL_ADDR -1ULL /* addr is not valid */
8687

87-
struct iomap_page_ops;
88+
struct iomap_folio_ops;
8889

8990
struct iomap {
9091
u64 addr; /* disk offset of mapping, bytes */
@@ -96,7 +97,7 @@ struct iomap {
9697
struct dax_device *dax_dev; /* dax_dev for dax operations */
9798
void *inline_data;
9899
void *private; /* filesystem private */
99-
const struct iomap_page_ops *page_ops;
100+
const struct iomap_folio_ops *folio_ops;
100101
u64 validity_cookie; /* used with .iomap_valid() */
101102
};
102103

@@ -124,19 +125,20 @@ static inline bool iomap_inline_data_valid(const struct iomap *iomap)
124125
}
125126

126127
/*
127-
* When a filesystem sets page_ops in an iomap mapping it returns, page_prepare
128-
* and page_done will be called for each page written to. This only applies to
129-
* buffered writes as unbuffered writes will not typically have pages
128+
* When a filesystem sets folio_ops in an iomap mapping it returns, get_folio
129+
* and put_folio will be called for each folio written to. This only applies
130+
* to buffered writes as unbuffered writes will not typically have folios
130131
* associated with them.
131132
*
132-
* When page_prepare succeeds, page_done will always be called to do any
133-
* cleanup work necessary. In that page_done call, @page will be NULL if the
134-
* associated page could not be obtained.
133+
* When get_folio succeeds, put_folio will always be called to do any
134+
* cleanup work necessary. put_folio is responsible for unlocking and putting
135+
* @folio.
135136
*/
136-
struct iomap_page_ops {
137-
int (*page_prepare)(struct inode *inode, loff_t pos, unsigned len);
138-
void (*page_done)(struct inode *inode, loff_t pos, unsigned copied,
139-
struct page *page);
137+
struct iomap_folio_ops {
138+
struct folio *(*get_folio)(struct iomap_iter *iter, loff_t pos,
139+
unsigned len);
140+
void (*put_folio)(struct inode *inode, loff_t pos, unsigned copied,
141+
struct folio *folio);
140142

141143
/*
142144
* Check that the cached iomap still maps correctly to the filesystem's
@@ -259,6 +261,7 @@ int iomap_file_buffered_write_punch_delalloc(struct inode *inode,
259261
int iomap_read_folio(struct folio *folio, const struct iomap_ops *ops);
260262
void iomap_readahead(struct readahead_control *, const struct iomap_ops *ops);
261263
bool iomap_is_partially_uptodate(struct folio *, size_t from, size_t count);
264+
struct folio *iomap_get_folio(struct iomap_iter *iter, loff_t pos);
262265
bool iomap_release_folio(struct folio *folio, gfp_t gfp_flags);
263266
void iomap_invalidate_folio(struct folio *folio, size_t offset, size_t len);
264267
int iomap_file_unshare(struct inode *inode, loff_t pos, loff_t len,

0 commit comments

Comments
 (0)