Skip to content

Commit a01b8f2

Browse files
committed
iomap: Allocate ifs in ->write_begin() early
We dont need to allocate an ifs in ->write_begin() for writes where the position and length completely overlap with the given folio. Therefore, such cases are skipped. Currently when the folio is uptodate, we only allocate ifs at writeback time (in iomap_writepage_map()). This is ok until now, but when we are going to add support for per-block dirty state bitmap in ifs, this could cause some performance degradation. The reason is that if we don't allocate ifs during ->write_begin(), then we will never mark the necessary dirty bits in ->write_end() call. And we will have to mark all the bits as dirty at the writeback time, that could cause the same write amplification and performance problems as it is now. Signed-off-by: Ritesh Harjani (IBM) <[email protected]> Reviewed-by: Darrick J. Wong <[email protected]> Reviewed-by: Christoph Hellwig <[email protected]>
1 parent 7f79d85 commit a01b8f2

File tree

1 file changed

+11
-2
lines changed

1 file changed

+11
-2
lines changed

fs/iomap/buffered-io.c

Lines changed: 11 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -557,14 +557,23 @@ static int __iomap_write_begin(const struct iomap_iter *iter, loff_t pos,
557557
size_t from = offset_in_folio(folio, pos), to = from + len;
558558
size_t poff, plen;
559559

560-
if (folio_test_uptodate(folio))
560+
/*
561+
* If the write completely overlaps the current folio, then
562+
* entire folio will be dirtied so there is no need for
563+
* per-block state tracking structures to be attached to this folio.
564+
*/
565+
if (pos <= folio_pos(folio) &&
566+
pos + len >= folio_pos(folio) + folio_size(folio))
561567
return 0;
562-
folio_clear_error(folio);
563568

564569
ifs = ifs_alloc(iter->inode, folio, iter->flags);
565570
if ((iter->flags & IOMAP_NOWAIT) && !ifs && nr_blocks > 1)
566571
return -EAGAIN;
567572

573+
if (folio_test_uptodate(folio))
574+
return 0;
575+
folio_clear_error(folio);
576+
568577
do {
569578
iomap_adjust_read_range(iter->inode, folio, &block_start,
570579
block_end - block_start, &poff, &plen);

0 commit comments

Comments
 (0)