2019-07-15 23:50:59 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2010 Red Hat, Inc.
|
2019-10-18 04:12:15 +08:00
|
|
|
* Copyright (C) 2016-2019 Christoph Hellwig.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/compiler.h>
|
|
|
|
#include <linux/fs.h>
|
|
|
|
#include <linux/iomap.h>
|
|
|
|
#include <linux/pagemap.h>
|
|
|
|
#include <linux/uio.h>
|
|
|
|
#include <linux/buffer_head.h>
|
|
|
|
#include <linux/dax.h>
|
|
|
|
#include <linux/writeback.h>
|
2019-10-18 04:12:15 +08:00
|
|
|
#include <linux/list_sort.h>
|
2019-07-15 23:50:59 +08:00
|
|
|
#include <linux/swap.h>
|
|
|
|
#include <linux/bio.h>
|
|
|
|
#include <linux/sched/signal.h>
|
|
|
|
#include <linux/migrate.h>
|
2019-10-18 04:12:13 +08:00
|
|
|
#include "trace.h"
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
#include "../internal.h"
|
|
|
|
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
#define IOEND_BATCH_SIZE 4096
|
|
|
|
|
2019-10-18 04:12:19 +08:00
|
|
|
/*
|
2021-04-28 10:56:17 +08:00
|
|
|
* Structure allocated for each folio when block size < folio size
|
|
|
|
* to track sub-folio uptodate status and I/O completions.
|
2019-10-18 04:12:19 +08:00
|
|
|
*/
|
|
|
|
struct iomap_page {
|
2020-09-21 23:58:40 +08:00
|
|
|
atomic_t read_bytes_pending;
|
2020-09-21 23:58:41 +08:00
|
|
|
atomic_t write_bytes_pending;
|
2019-12-05 01:33:52 +08:00
|
|
|
spinlock_t uptodate_lock;
|
2020-09-21 23:58:40 +08:00
|
|
|
unsigned long uptodate[];
|
2019-10-18 04:12:19 +08:00
|
|
|
};
|
|
|
|
|
2021-04-28 10:56:17 +08:00
|
|
|
static inline struct iomap_page *to_iomap_page(struct folio *folio)
|
2019-10-18 04:12:19 +08:00
|
|
|
{
|
2021-04-28 10:56:17 +08:00
|
|
|
if (folio_test_private(folio))
|
|
|
|
return folio_get_private(folio);
|
2019-10-18 04:12:19 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-10-18 04:12:15 +08:00
|
|
|
static struct bio_set iomap_ioend_bioset;
|
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
static struct iomap_page *
|
2022-06-24 01:51:47 +08:00
|
|
|
iomap_page_create(struct inode *inode, struct folio *folio, unsigned int flags)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 10:56:17 +08:00
|
|
|
struct iomap_page *iop = to_iomap_page(folio);
|
2021-04-28 11:12:52 +08:00
|
|
|
unsigned int nr_blocks = i_blocks_per_folio(inode, folio);
|
2022-06-24 01:51:47 +08:00
|
|
|
gfp_t gfp;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2020-09-21 23:58:40 +08:00
|
|
|
if (iop || nr_blocks <= 1)
|
2019-07-15 23:50:59 +08:00
|
|
|
return iop;
|
|
|
|
|
2022-06-24 01:51:47 +08:00
|
|
|
if (flags & IOMAP_NOWAIT)
|
|
|
|
gfp = GFP_NOWAIT;
|
|
|
|
else
|
|
|
|
gfp = GFP_NOFS | __GFP_NOFAIL;
|
|
|
|
|
2020-09-21 23:58:40 +08:00
|
|
|
iop = kzalloc(struct_size(iop, uptodate, BITS_TO_LONGS(nr_blocks)),
|
2022-06-24 01:51:47 +08:00
|
|
|
gfp);
|
|
|
|
if (iop) {
|
|
|
|
spin_lock_init(&iop->uptodate_lock);
|
|
|
|
if (folio_test_uptodate(folio))
|
|
|
|
bitmap_fill(iop->uptodate, nr_blocks);
|
|
|
|
folio_attach_private(folio, iop);
|
|
|
|
}
|
2019-07-15 23:50:59 +08:00
|
|
|
return iop;
|
|
|
|
}
|
|
|
|
|
2021-04-28 11:22:22 +08:00
|
|
|
static void iomap_page_release(struct folio *folio)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 11:22:22 +08:00
|
|
|
struct iomap_page *iop = folio_detach_private(folio);
|
|
|
|
struct inode *inode = folio->mapping->host;
|
|
|
|
unsigned int nr_blocks = i_blocks_per_folio(inode, folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (!iop)
|
|
|
|
return;
|
2020-09-21 23:58:40 +08:00
|
|
|
WARN_ON_ONCE(atomic_read(&iop->read_bytes_pending));
|
2020-09-21 23:58:41 +08:00
|
|
|
WARN_ON_ONCE(atomic_read(&iop->write_bytes_pending));
|
2020-09-21 23:58:40 +08:00
|
|
|
WARN_ON_ONCE(bitmap_full(iop->uptodate, nr_blocks) !=
|
2021-04-28 11:22:22 +08:00
|
|
|
folio_test_uptodate(folio));
|
2019-07-15 23:50:59 +08:00
|
|
|
kfree(iop);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2021-04-28 20:20:48 +08:00
|
|
|
* Calculate the range inside the folio that we actually need to read.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
2021-04-28 20:20:48 +08:00
|
|
|
static void iomap_adjust_read_range(struct inode *inode, struct folio *folio,
|
|
|
|
loff_t *pos, loff_t length, size_t *offp, size_t *lenp)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 20:20:48 +08:00
|
|
|
struct iomap_page *iop = to_iomap_page(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
loff_t orig_pos = *pos;
|
|
|
|
loff_t isize = i_size_read(inode);
|
|
|
|
unsigned block_bits = inode->i_blkbits;
|
|
|
|
unsigned block_size = (1 << block_bits);
|
2021-04-28 20:20:48 +08:00
|
|
|
size_t poff = offset_in_folio(folio, *pos);
|
|
|
|
size_t plen = min_t(loff_t, folio_size(folio) - poff, length);
|
2019-07-15 23:50:59 +08:00
|
|
|
unsigned first = poff >> block_bits;
|
|
|
|
unsigned last = (poff + plen - 1) >> block_bits;
|
|
|
|
|
|
|
|
/*
|
2021-08-03 05:46:31 +08:00
|
|
|
* If the block size is smaller than the page size, we need to check the
|
2019-07-15 23:50:59 +08:00
|
|
|
* per-block uptodate status and adjust the offset and length if needed
|
|
|
|
* to avoid reading in already uptodate ranges.
|
|
|
|
*/
|
|
|
|
if (iop) {
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
/* move forward for each leading block marked uptodate */
|
|
|
|
for (i = first; i <= last; i++) {
|
|
|
|
if (!test_bit(i, iop->uptodate))
|
|
|
|
break;
|
|
|
|
*pos += block_size;
|
|
|
|
poff += block_size;
|
|
|
|
plen -= block_size;
|
|
|
|
first++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* truncate len if we find any trailing uptodate block(s) */
|
|
|
|
for ( ; i <= last; i++) {
|
|
|
|
if (test_bit(i, iop->uptodate)) {
|
|
|
|
plen -= (last - i + 1) * block_size;
|
|
|
|
last = i - 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2021-08-03 05:46:31 +08:00
|
|
|
* If the extent spans the block that contains the i_size, we need to
|
2019-07-15 23:50:59 +08:00
|
|
|
* handle both halves separately so that we properly zero data in the
|
|
|
|
* page cache for blocks that are entirely outside of i_size.
|
|
|
|
*/
|
|
|
|
if (orig_pos <= isize && orig_pos + length > isize) {
|
2021-04-28 20:20:48 +08:00
|
|
|
unsigned end = offset_in_folio(folio, isize - 1) >> block_bits;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (first <= end && last > end)
|
|
|
|
plen -= (last - end) * block_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
*offp = poff;
|
|
|
|
*lenp = plen;
|
|
|
|
}
|
|
|
|
|
2021-04-28 20:20:48 +08:00
|
|
|
static void iomap_iop_set_range_uptodate(struct folio *folio,
|
|
|
|
struct iomap_page *iop, size_t off, size_t len)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 20:20:48 +08:00
|
|
|
struct inode *inode = folio->mapping->host;
|
2019-07-15 23:50:59 +08:00
|
|
|
unsigned first = off >> inode->i_blkbits;
|
|
|
|
unsigned last = (off + len - 1) >> inode->i_blkbits;
|
2019-12-05 01:33:52 +08:00
|
|
|
unsigned long flags;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2019-12-05 01:33:52 +08:00
|
|
|
spin_lock_irqsave(&iop->uptodate_lock, flags);
|
2020-09-21 23:58:40 +08:00
|
|
|
bitmap_set(iop->uptodate, first, last - first + 1);
|
2021-04-28 20:20:48 +08:00
|
|
|
if (bitmap_full(iop->uptodate, i_blocks_per_folio(inode, folio)))
|
|
|
|
folio_mark_uptodate(folio);
|
2019-12-05 01:33:52 +08:00
|
|
|
spin_unlock_irqrestore(&iop->uptodate_lock, flags);
|
|
|
|
}
|
|
|
|
|
2021-04-28 20:20:48 +08:00
|
|
|
static void iomap_set_range_uptodate(struct folio *folio,
|
|
|
|
struct iomap_page *iop, size_t off, size_t len)
|
2019-12-05 01:33:52 +08:00
|
|
|
{
|
2021-04-28 20:16:50 +08:00
|
|
|
if (iop)
|
2021-04-28 20:20:48 +08:00
|
|
|
iomap_iop_set_range_uptodate(folio, iop, off, len);
|
2019-12-05 01:33:52 +08:00
|
|
|
else
|
2021-04-28 20:20:48 +08:00
|
|
|
folio_mark_uptodate(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
2021-01-02 05:53:26 +08:00
|
|
|
static void iomap_finish_folio_read(struct folio *folio, size_t offset,
|
|
|
|
size_t len, int error)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 10:56:17 +08:00
|
|
|
struct iomap_page *iop = to_iomap_page(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (unlikely(error)) {
|
2021-01-02 05:53:26 +08:00
|
|
|
folio_clear_uptodate(folio);
|
|
|
|
folio_set_error(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
} else {
|
2021-04-28 20:20:48 +08:00
|
|
|
iomap_set_range_uptodate(folio, iop, offset, len);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
2021-01-02 05:53:26 +08:00
|
|
|
if (!iop || atomic_sub_and_test(len, &iop->read_bytes_pending))
|
|
|
|
folio_unlock(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
2021-01-02 05:53:26 +08:00
|
|
|
static void iomap_read_end_io(struct bio *bio)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
|
|
|
int error = blk_status_to_errno(bio->bi_status);
|
2021-01-02 05:53:26 +08:00
|
|
|
struct folio_iter fi;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-01-02 05:53:26 +08:00
|
|
|
bio_for_each_folio_all(fi, bio)
|
|
|
|
iomap_finish_folio_read(fi.folio, fi.offset, fi.length, error);
|
2019-07-15 23:50:59 +08:00
|
|
|
bio_put(bio);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct iomap_readpage_ctx {
|
2021-04-28 21:39:51 +08:00
|
|
|
struct folio *cur_folio;
|
|
|
|
bool cur_folio_in_bio;
|
2019-07-15 23:50:59 +08:00
|
|
|
struct bio *bio;
|
2020-06-02 12:47:34 +08:00
|
|
|
struct readahead_control *rac;
|
2019-07-15 23:50:59 +08:00
|
|
|
};
|
|
|
|
|
2021-11-25 02:15:47 +08:00
|
|
|
/**
|
|
|
|
* iomap_read_inline_data - copy inline data into the page cache
|
|
|
|
* @iter: iteration structure
|
2021-07-24 11:24:50 +08:00
|
|
|
* @folio: folio to copy to
|
2021-11-25 02:15:47 +08:00
|
|
|
*
|
2021-07-24 11:24:50 +08:00
|
|
|
* Copy the inline data in @iter into @folio and zero out the rest of the folio.
|
2021-11-25 02:15:47 +08:00
|
|
|
* Only a single IOMAP_INLINE extent is allowed at the end of each file.
|
|
|
|
* Returns zero for success to complete the read, or the usual negative errno.
|
|
|
|
*/
|
|
|
|
static int iomap_read_inline_data(const struct iomap_iter *iter,
|
2021-07-24 11:24:50 +08:00
|
|
|
struct folio *folio)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 20:16:50 +08:00
|
|
|
struct iomap_page *iop;
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *iomap = iomap_iter_srcmap(iter);
|
2021-08-11 09:33:14 +08:00
|
|
|
size_t size = i_size_read(iter->inode) - iomap->offset;
|
2021-08-03 05:45:57 +08:00
|
|
|
size_t poff = offset_in_page(iomap->offset);
|
2021-04-28 20:20:48 +08:00
|
|
|
size_t offset = offset_in_folio(folio, iomap->offset);
|
2019-07-15 23:50:59 +08:00
|
|
|
void *addr;
|
|
|
|
|
2021-07-24 11:24:50 +08:00
|
|
|
if (folio_test_uptodate(folio))
|
2021-11-25 02:15:47 +08:00
|
|
|
return 0;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-08-05 11:07:34 +08:00
|
|
|
if (WARN_ON_ONCE(size > PAGE_SIZE - poff))
|
|
|
|
return -EIO;
|
2021-08-04 00:38:22 +08:00
|
|
|
if (WARN_ON_ONCE(size > PAGE_SIZE -
|
|
|
|
offset_in_page(iomap->inline_data)))
|
|
|
|
return -EIO;
|
|
|
|
if (WARN_ON_ONCE(size > iomap->length))
|
|
|
|
return -EIO;
|
2021-04-28 20:20:48 +08:00
|
|
|
if (offset > 0)
|
2022-06-24 01:51:47 +08:00
|
|
|
iop = iomap_page_create(iter->inode, folio, iter->flags);
|
2021-04-28 20:16:50 +08:00
|
|
|
else
|
|
|
|
iop = to_iomap_page(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-07-24 11:24:50 +08:00
|
|
|
addr = kmap_local_folio(folio, offset);
|
2019-07-15 23:50:59 +08:00
|
|
|
memcpy(addr, iomap->inline_data, size);
|
2021-08-03 05:45:57 +08:00
|
|
|
memset(addr + size, 0, PAGE_SIZE - poff - size);
|
2021-08-05 11:07:33 +08:00
|
|
|
kunmap_local(addr);
|
2021-04-28 20:20:48 +08:00
|
|
|
iomap_set_range_uptodate(folio, iop, offset, PAGE_SIZE - poff);
|
2021-11-25 02:15:47 +08:00
|
|
|
return 0;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
2021-08-11 09:33:16 +08:00
|
|
|
static inline bool iomap_block_needs_zeroing(const struct iomap_iter *iter,
|
2021-08-11 09:33:14 +08:00
|
|
|
loff_t pos)
|
2019-10-18 04:12:12 +08:00
|
|
|
{
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *srcmap = iomap_iter_srcmap(iter);
|
2021-08-11 09:33:14 +08:00
|
|
|
|
|
|
|
return srcmap->type != IOMAP_MAPPED ||
|
|
|
|
(srcmap->flags & IOMAP_F_NEW) ||
|
|
|
|
pos >= i_size_read(iter->inode);
|
2019-10-18 04:12:12 +08:00
|
|
|
}
|
|
|
|
|
2021-08-11 09:33:16 +08:00
|
|
|
static loff_t iomap_readpage_iter(const struct iomap_iter *iter,
|
2021-08-11 09:33:08 +08:00
|
|
|
struct iomap_readpage_ctx *ctx, loff_t offset)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *iomap = &iter->iomap;
|
2021-08-11 09:33:08 +08:00
|
|
|
loff_t pos = iter->pos + offset;
|
|
|
|
loff_t length = iomap_length(iter) - offset;
|
2021-04-28 21:39:51 +08:00
|
|
|
struct folio *folio = ctx->cur_folio;
|
2021-07-16 00:58:05 +08:00
|
|
|
struct iomap_page *iop;
|
2019-07-15 23:50:59 +08:00
|
|
|
loff_t orig_pos = pos;
|
2021-04-28 20:20:48 +08:00
|
|
|
size_t poff, plen;
|
2019-07-15 23:50:59 +08:00
|
|
|
sector_t sector;
|
|
|
|
|
2021-11-25 02:15:47 +08:00
|
|
|
if (iomap->type == IOMAP_INLINE)
|
2021-07-24 11:24:50 +08:00
|
|
|
return iomap_read_inline_data(iter, folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
/* zero post-eof blocks as the page may be mapped */
|
2022-06-24 01:51:47 +08:00
|
|
|
iop = iomap_page_create(iter->inode, folio, iter->flags);
|
2021-04-28 20:20:48 +08:00
|
|
|
iomap_adjust_read_range(iter->inode, folio, &pos, length, &poff, &plen);
|
2019-07-15 23:50:59 +08:00
|
|
|
if (plen == 0)
|
|
|
|
goto done;
|
|
|
|
|
2021-08-11 09:33:14 +08:00
|
|
|
if (iomap_block_needs_zeroing(iter, pos)) {
|
2021-04-28 20:20:48 +08:00
|
|
|
folio_zero_range(folio, poff, plen);
|
|
|
|
iomap_set_range_uptodate(folio, iop, poff, plen);
|
2019-07-15 23:50:59 +08:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2021-04-28 21:39:51 +08:00
|
|
|
ctx->cur_folio_in_bio = true;
|
2020-09-21 23:58:40 +08:00
|
|
|
if (iop)
|
|
|
|
atomic_add(plen, &iop->read_bytes_pending);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
sector = iomap_sector(iomap, pos);
|
2021-08-03 05:43:43 +08:00
|
|
|
if (!ctx->bio ||
|
|
|
|
bio_end_sector(ctx->bio) != sector ||
|
2021-04-28 20:20:48 +08:00
|
|
|
!bio_add_folio(ctx->bio, folio, plen, poff)) {
|
2021-04-28 21:39:51 +08:00
|
|
|
gfp_t gfp = mapping_gfp_constraint(folio->mapping, GFP_KERNEL);
|
2020-04-03 00:08:53 +08:00
|
|
|
gfp_t orig_gfp = gfp;
|
2021-01-29 12:38:57 +08:00
|
|
|
unsigned int nr_vecs = DIV_ROUND_UP(length, PAGE_SIZE);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (ctx->bio)
|
|
|
|
submit_bio(ctx->bio);
|
|
|
|
|
2020-06-02 12:47:34 +08:00
|
|
|
if (ctx->rac) /* same as readahead_gfp_mask */
|
2019-07-15 23:50:59 +08:00
|
|
|
gfp |= __GFP_NORETRY | __GFP_NOWARN;
|
2022-01-24 17:11:05 +08:00
|
|
|
ctx->bio = bio_alloc(iomap->bdev, bio_max_segs(nr_vecs),
|
|
|
|
REQ_OP_READ, gfp);
|
2020-04-03 00:08:53 +08:00
|
|
|
/*
|
|
|
|
* If the bio_alloc fails, try it again for a single page to
|
|
|
|
* avoid having to deal with partial page reads. This emulates
|
2022-04-29 23:47:39 +08:00
|
|
|
* what do_mpage_read_folio does.
|
2020-04-03 00:08:53 +08:00
|
|
|
*/
|
2022-01-24 17:11:05 +08:00
|
|
|
if (!ctx->bio) {
|
|
|
|
ctx->bio = bio_alloc(iomap->bdev, 1, REQ_OP_READ,
|
|
|
|
orig_gfp);
|
|
|
|
}
|
2020-06-02 12:47:34 +08:00
|
|
|
if (ctx->rac)
|
2019-07-15 23:50:59 +08:00
|
|
|
ctx->bio->bi_opf |= REQ_RAHEAD;
|
|
|
|
ctx->bio->bi_iter.bi_sector = sector;
|
|
|
|
ctx->bio->bi_end_io = iomap_read_end_io;
|
2021-04-28 20:20:48 +08:00
|
|
|
bio_add_folio(ctx->bio, folio, plen, poff);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
2021-04-28 20:20:48 +08:00
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
done:
|
|
|
|
/*
|
|
|
|
* Move the caller beyond our range so that it keeps making progress.
|
2021-08-03 05:46:31 +08:00
|
|
|
* For that, we have to include any leading non-uptodate ranges, but
|
2019-07-15 23:50:59 +08:00
|
|
|
* we can skip trailing ones as they will be handled in the next
|
|
|
|
* iteration.
|
|
|
|
*/
|
|
|
|
return pos - orig_pos + plen;
|
|
|
|
}
|
|
|
|
|
2022-04-29 20:54:32 +08:00
|
|
|
int iomap_read_folio(struct folio *folio, const struct iomap_ops *ops)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:08 +08:00
|
|
|
struct iomap_iter iter = {
|
2021-04-28 21:39:51 +08:00
|
|
|
.inode = folio->mapping->host,
|
|
|
|
.pos = folio_pos(folio),
|
|
|
|
.len = folio_size(folio),
|
2021-08-11 09:33:08 +08:00
|
|
|
};
|
|
|
|
struct iomap_readpage_ctx ctx = {
|
2021-04-28 21:39:51 +08:00
|
|
|
.cur_folio = folio,
|
2021-08-11 09:33:08 +08:00
|
|
|
};
|
|
|
|
int ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-04-28 21:39:51 +08:00
|
|
|
trace_iomap_readpage(iter.inode, 1);
|
2019-10-18 04:12:13 +08:00
|
|
|
|
2021-08-11 09:33:08 +08:00
|
|
|
while ((ret = iomap_iter(&iter, ops)) > 0)
|
|
|
|
iter.processed = iomap_readpage_iter(&iter, &ctx, 0);
|
|
|
|
|
|
|
|
if (ret < 0)
|
2021-04-28 21:39:51 +08:00
|
|
|
folio_set_error(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (ctx.bio) {
|
|
|
|
submit_bio(ctx.bio);
|
2021-04-28 21:39:51 +08:00
|
|
|
WARN_ON_ONCE(!ctx.cur_folio_in_bio);
|
2019-07-15 23:50:59 +08:00
|
|
|
} else {
|
2021-04-28 21:39:51 +08:00
|
|
|
WARN_ON_ONCE(ctx.cur_folio_in_bio);
|
|
|
|
folio_unlock(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2022-04-29 22:40:40 +08:00
|
|
|
* Just like mpage_readahead and block_read_full_folio, we always
|
2022-04-29 20:54:32 +08:00
|
|
|
* return 0 and just set the folio error flag on errors. This
|
2021-08-03 05:46:31 +08:00
|
|
|
* should be cleaned up throughout the stack eventually.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
|
|
|
return 0;
|
|
|
|
}
|
2022-04-29 20:54:32 +08:00
|
|
|
EXPORT_SYMBOL_GPL(iomap_read_folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-08-11 09:33:16 +08:00
|
|
|
static loff_t iomap_readahead_iter(const struct iomap_iter *iter,
|
2021-08-11 09:33:08 +08:00
|
|
|
struct iomap_readpage_ctx *ctx)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:08 +08:00
|
|
|
loff_t length = iomap_length(iter);
|
2019-07-15 23:50:59 +08:00
|
|
|
loff_t done, ret;
|
|
|
|
|
|
|
|
for (done = 0; done < length; done += ret) {
|
2021-04-28 21:39:51 +08:00
|
|
|
if (ctx->cur_folio &&
|
|
|
|
offset_in_folio(ctx->cur_folio, iter->pos + done) == 0) {
|
|
|
|
if (!ctx->cur_folio_in_bio)
|
|
|
|
folio_unlock(ctx->cur_folio);
|
|
|
|
ctx->cur_folio = NULL;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
2021-04-28 21:39:51 +08:00
|
|
|
if (!ctx->cur_folio) {
|
|
|
|
ctx->cur_folio = readahead_folio(ctx->rac);
|
|
|
|
ctx->cur_folio_in_bio = false;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
2021-08-11 09:33:08 +08:00
|
|
|
ret = iomap_readpage_iter(iter, ctx, done);
|
2021-11-18 09:59:01 +08:00
|
|
|
if (ret <= 0)
|
|
|
|
return ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return done;
|
|
|
|
}
|
|
|
|
|
2020-06-02 12:47:34 +08:00
|
|
|
/**
|
|
|
|
* iomap_readahead - Attempt to read pages from a file.
|
|
|
|
* @rac: Describes the pages to be read.
|
|
|
|
* @ops: The operations vector for the filesystem.
|
|
|
|
*
|
|
|
|
* This function is for filesystems to call to implement their readahead
|
|
|
|
* address_space operation.
|
|
|
|
*
|
|
|
|
* Context: The @ops callbacks may submit I/O (eg to read the addresses of
|
|
|
|
* blocks from disc), and may wait for it. The caller may be trying to
|
|
|
|
* access a different page, and so sleeping excessively should be avoided.
|
|
|
|
* It may allocate memory, but should avoid costly allocations. This
|
|
|
|
* function is called with memalloc_nofs set, so allocations will not cause
|
|
|
|
* the filesystem to be reentered.
|
|
|
|
*/
|
|
|
|
void iomap_readahead(struct readahead_control *rac, const struct iomap_ops *ops)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:08 +08:00
|
|
|
struct iomap_iter iter = {
|
|
|
|
.inode = rac->mapping->host,
|
|
|
|
.pos = readahead_pos(rac),
|
|
|
|
.len = readahead_length(rac),
|
|
|
|
};
|
2019-07-15 23:50:59 +08:00
|
|
|
struct iomap_readpage_ctx ctx = {
|
2020-06-02 12:47:34 +08:00
|
|
|
.rac = rac,
|
2019-07-15 23:50:59 +08:00
|
|
|
};
|
|
|
|
|
2021-08-11 09:33:08 +08:00
|
|
|
trace_iomap_readahead(rac->mapping->host, readahead_count(rac));
|
2019-10-18 04:12:13 +08:00
|
|
|
|
2021-08-11 09:33:08 +08:00
|
|
|
while (iomap_iter(&iter, ops) > 0)
|
|
|
|
iter.processed = iomap_readahead_iter(&iter, &ctx);
|
2020-06-02 12:47:34 +08:00
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
if (ctx.bio)
|
|
|
|
submit_bio(ctx.bio);
|
2021-04-28 21:39:51 +08:00
|
|
|
if (ctx.cur_folio) {
|
|
|
|
if (!ctx.cur_folio_in_bio)
|
|
|
|
folio_unlock(ctx.cur_folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
}
|
2020-06-02 12:47:34 +08:00
|
|
|
EXPORT_SYMBOL_GPL(iomap_readahead);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
/*
|
2022-02-10 04:21:27 +08:00
|
|
|
* iomap_is_partially_uptodate checks whether blocks within a folio are
|
2019-07-15 23:50:59 +08:00
|
|
|
* uptodate or not.
|
|
|
|
*
|
2022-02-10 04:21:27 +08:00
|
|
|
* Returns true if all blocks which correspond to the specified part
|
|
|
|
* of the folio are uptodate.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
2022-02-10 04:21:27 +08:00
|
|
|
bool iomap_is_partially_uptodate(struct folio *folio, size_t from, size_t count)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 10:56:17 +08:00
|
|
|
struct iomap_page *iop = to_iomap_page(folio);
|
2022-02-10 04:21:27 +08:00
|
|
|
struct inode *inode = folio->mapping->host;
|
|
|
|
unsigned first, last, i;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2022-02-10 04:21:27 +08:00
|
|
|
if (!iop)
|
|
|
|
return false;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2022-03-05 03:05:23 +08:00
|
|
|
/* Caller's range may extend past the end of this folio */
|
|
|
|
count = min(folio_size(folio) - from, count);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2022-03-05 03:05:23 +08:00
|
|
|
/* First and last blocks in range within folio */
|
2019-07-15 23:50:59 +08:00
|
|
|
first = from >> inode->i_blkbits;
|
2022-03-05 03:05:23 +08:00
|
|
|
last = (from + count - 1) >> inode->i_blkbits;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2022-02-10 04:21:27 +08:00
|
|
|
for (i = first; i <= last; i++)
|
|
|
|
if (!test_bit(i, iop->uptodate))
|
|
|
|
return false;
|
|
|
|
return true;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_is_partially_uptodate);
|
|
|
|
|
2022-05-01 11:01:08 +08:00
|
|
|
bool iomap_release_folio(struct folio *folio, gfp_t gfp_flags)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2022-05-01 11:01:08 +08:00
|
|
|
trace_iomap_release_folio(folio->mapping->host, folio_pos(folio),
|
2021-04-28 19:51:36 +08:00
|
|
|
folio_size(folio));
|
2019-10-18 04:12:13 +08:00
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
/*
|
2022-05-01 11:01:08 +08:00
|
|
|
* mm accommodates an old ext3 case where clean folios might
|
|
|
|
* not have had the dirty bit cleared. Thus, it can send actual
|
|
|
|
* dirty folios to ->release_folio() via shrink_active_list();
|
|
|
|
* skip those here.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
2021-04-28 19:51:36 +08:00
|
|
|
if (folio_test_dirty(folio) || folio_test_writeback(folio))
|
2022-05-01 11:01:08 +08:00
|
|
|
return false;
|
2021-04-28 11:22:22 +08:00
|
|
|
iomap_page_release(folio);
|
2022-05-01 11:01:08 +08:00
|
|
|
return true;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
2022-05-01 11:01:08 +08:00
|
|
|
EXPORT_SYMBOL_GPL(iomap_release_folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-04-28 19:51:36 +08:00
|
|
|
void iomap_invalidate_folio(struct folio *folio, size_t offset, size_t len)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2022-02-10 04:21:33 +08:00
|
|
|
trace_iomap_invalidate_folio(folio->mapping->host,
|
2022-02-10 04:21:22 +08:00
|
|
|
folio_pos(folio) + offset, len);
|
2019-10-18 04:12:13 +08:00
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
/*
|
2021-01-13 23:48:49 +08:00
|
|
|
* If we're invalidating the entire folio, clear the dirty state
|
|
|
|
* from it and release it to avoid unnecessary buildup of the LRU.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
2021-04-28 19:51:36 +08:00
|
|
|
if (offset == 0 && len == folio_size(folio)) {
|
|
|
|
WARN_ON_ONCE(folio_test_writeback(folio));
|
|
|
|
folio_cancel_dirty(folio);
|
2021-04-28 11:22:22 +08:00
|
|
|
iomap_page_release(folio);
|
2021-01-13 23:48:49 +08:00
|
|
|
} else if (folio_test_large(folio)) {
|
|
|
|
/* Must release the iop so the page can be split */
|
|
|
|
WARN_ON_ONCE(!folio_test_uptodate(folio) &&
|
|
|
|
folio_test_dirty(folio));
|
|
|
|
iomap_page_release(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
}
|
2021-04-28 19:51:36 +08:00
|
|
|
EXPORT_SYMBOL_GPL(iomap_invalidate_folio);
|
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
static void
|
|
|
|
iomap_write_failed(struct inode *inode, loff_t pos, unsigned len)
|
|
|
|
{
|
|
|
|
loff_t i_size = i_size_read(inode);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Only truncate newly allocated pages beyoned EOF, even if the
|
|
|
|
* write started inside the existing inode size.
|
|
|
|
*/
|
|
|
|
if (pos + len > i_size)
|
2022-05-06 09:19:13 +08:00
|
|
|
truncate_pagecache_range(inode, max(pos, i_size),
|
|
|
|
pos + len - 1);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
2021-04-28 20:20:48 +08:00
|
|
|
static int iomap_read_folio_sync(loff_t block_start, struct folio *folio,
|
|
|
|
size_t poff, size_t plen, const struct iomap *iomap)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
|
|
|
struct bio_vec bvec;
|
|
|
|
struct bio bio;
|
|
|
|
|
2022-01-24 17:11:06 +08:00
|
|
|
bio_init(&bio, iomap->bdev, &bvec, 1, REQ_OP_READ);
|
2019-07-15 23:50:59 +08:00
|
|
|
bio.bi_iter.bi_sector = iomap_sector(iomap, block_start);
|
2021-04-28 20:20:48 +08:00
|
|
|
bio_add_folio(&bio, folio, plen, poff);
|
2019-07-15 23:50:59 +08:00
|
|
|
return submit_bio_wait(&bio);
|
|
|
|
}
|
|
|
|
|
2021-08-11 09:33:16 +08:00
|
|
|
static int __iomap_write_begin(const struct iomap_iter *iter, loff_t pos,
|
2021-05-02 23:33:08 +08:00
|
|
|
size_t len, struct folio *folio)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *srcmap = iomap_iter_srcmap(iter);
|
2022-06-24 01:51:47 +08:00
|
|
|
struct iomap_page *iop;
|
2021-08-11 09:33:14 +08:00
|
|
|
loff_t block_size = i_blocksize(iter->inode);
|
2020-09-10 23:38:06 +08:00
|
|
|
loff_t block_start = round_down(pos, block_size);
|
|
|
|
loff_t block_end = round_up(pos + len, block_size);
|
2022-06-24 01:51:48 +08:00
|
|
|
unsigned int nr_blocks = i_blocks_per_folio(iter->inode, folio);
|
2021-04-28 20:20:48 +08:00
|
|
|
size_t from = offset_in_folio(folio, pos), to = from + len;
|
|
|
|
size_t poff, plen;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-04-28 20:20:48 +08:00
|
|
|
if (folio_test_uptodate(folio))
|
2019-07-15 23:50:59 +08:00
|
|
|
return 0;
|
2021-04-28 20:20:48 +08:00
|
|
|
folio_clear_error(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2022-06-24 01:51:47 +08:00
|
|
|
iop = iomap_page_create(iter->inode, folio, iter->flags);
|
2022-06-24 01:51:48 +08:00
|
|
|
if ((iter->flags & IOMAP_NOWAIT) && !iop && nr_blocks > 1)
|
|
|
|
return -EAGAIN;
|
2022-06-24 01:51:47 +08:00
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
do {
|
2021-04-28 20:20:48 +08:00
|
|
|
iomap_adjust_read_range(iter->inode, folio, &block_start,
|
2019-07-15 23:50:59 +08:00
|
|
|
block_end - block_start, &poff, &plen);
|
|
|
|
if (plen == 0)
|
|
|
|
break;
|
|
|
|
|
2021-08-11 09:33:14 +08:00
|
|
|
if (!(iter->flags & IOMAP_UNSHARE) &&
|
2019-10-19 07:42:50 +08:00
|
|
|
(from <= poff || from >= poff + plen) &&
|
2019-10-19 07:42:24 +08:00
|
|
|
(to <= poff || to >= poff + plen))
|
|
|
|
continue;
|
|
|
|
|
2021-08-11 09:33:14 +08:00
|
|
|
if (iomap_block_needs_zeroing(iter, block_start)) {
|
2021-08-11 09:33:14 +08:00
|
|
|
if (WARN_ON_ONCE(iter->flags & IOMAP_UNSHARE))
|
2019-10-19 07:42:50 +08:00
|
|
|
return -EIO;
|
2021-04-28 20:20:48 +08:00
|
|
|
folio_zero_segments(folio, poff, from, to, poff + plen);
|
2020-09-10 23:26:18 +08:00
|
|
|
} else {
|
2022-06-24 01:51:48 +08:00
|
|
|
int status;
|
|
|
|
|
|
|
|
if (iter->flags & IOMAP_NOWAIT)
|
|
|
|
return -EAGAIN;
|
|
|
|
|
|
|
|
status = iomap_read_folio_sync(block_start, folio,
|
2020-09-10 23:26:18 +08:00
|
|
|
poff, plen, srcmap);
|
|
|
|
if (status)
|
|
|
|
return status;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
2021-04-28 20:20:48 +08:00
|
|
|
iomap_set_range_uptodate(folio, iop, poff, plen);
|
2019-07-15 23:50:59 +08:00
|
|
|
} while ((block_start += plen) < block_end);
|
|
|
|
|
2019-10-19 07:42:24 +08:00
|
|
|
return 0;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
2021-08-11 09:33:16 +08:00
|
|
|
static int iomap_write_begin_inline(const struct iomap_iter *iter,
|
2021-05-02 23:33:08 +08:00
|
|
|
struct folio *folio)
|
2021-08-04 00:38:22 +08:00
|
|
|
{
|
|
|
|
/* needs more work for the tailpacking case; disable for now */
|
2021-08-11 09:33:14 +08:00
|
|
|
if (WARN_ON_ONCE(iomap_iter_srcmap(iter)->offset != 0))
|
2021-08-04 00:38:22 +08:00
|
|
|
return -EIO;
|
2021-07-24 11:24:50 +08:00
|
|
|
return iomap_read_inline_data(iter, folio);
|
2021-08-04 00:38:22 +08:00
|
|
|
}
|
|
|
|
|
2022-11-29 06:09:17 +08:00
|
|
|
static int iomap_write_begin(struct iomap_iter *iter, loff_t pos,
|
2021-05-02 23:33:08 +08:00
|
|
|
size_t len, struct folio **foliop)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:14 +08:00
|
|
|
const struct iomap_page_ops *page_ops = iter->iomap.page_ops;
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *srcmap = iomap_iter_srcmap(iter);
|
2021-11-04 02:05:47 +08:00
|
|
|
struct folio *folio;
|
2021-05-02 23:33:08 +08:00
|
|
|
unsigned fgp = FGP_LOCK | FGP_WRITE | FGP_CREAT | FGP_STABLE | FGP_NOFS;
|
2019-07-15 23:50:59 +08:00
|
|
|
int status = 0;
|
|
|
|
|
2022-06-24 01:51:48 +08:00
|
|
|
if (iter->flags & IOMAP_NOWAIT)
|
|
|
|
fgp |= FGP_NOWAIT;
|
|
|
|
|
2021-08-11 09:33:14 +08:00
|
|
|
BUG_ON(pos + len > iter->iomap.offset + iter->iomap.length);
|
|
|
|
if (srcmap != &iter->iomap)
|
2019-10-19 07:44:10 +08:00
|
|
|
BUG_ON(pos + len > srcmap->offset + srcmap->length);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (fatal_signal_pending(current))
|
|
|
|
return -EINTR;
|
|
|
|
|
2021-12-10 04:47:44 +08:00
|
|
|
if (!mapping_large_folio_support(iter->inode->i_mapping))
|
|
|
|
len = min_t(size_t, len, PAGE_SIZE - offset_in_page(pos));
|
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
if (page_ops && page_ops->page_prepare) {
|
2021-08-11 09:33:14 +08:00
|
|
|
status = page_ops->page_prepare(iter->inode, pos, len);
|
2019-07-15 23:50:59 +08:00
|
|
|
if (status)
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2021-05-02 23:33:08 +08:00
|
|
|
folio = __filemap_get_folio(iter->inode->i_mapping, pos >> PAGE_SHIFT,
|
|
|
|
fgp, mapping_gfp_mask(iter->inode->i_mapping));
|
|
|
|
if (!folio) {
|
2022-06-24 01:51:48 +08:00
|
|
|
status = (iter->flags & IOMAP_NOWAIT) ? -EAGAIN : -ENOMEM;
|
2019-07-15 23:50:59 +08:00
|
|
|
goto out_no_page;
|
|
|
|
}
|
2022-11-29 06:09:17 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Now we have a locked folio, before we do anything with it we need to
|
|
|
|
* check that the iomap we have cached is not stale. The inode extent
|
|
|
|
* mapping can change due to concurrent IO in flight (e.g.
|
|
|
|
* IOMAP_UNWRITTEN state can change and memory reclaim could have
|
|
|
|
* reclaimed a previously partially written page at this index after IO
|
|
|
|
* completion before this write reaches this file offset) and hence we
|
|
|
|
* could do the wrong thing here (zero a page range incorrectly or fail
|
|
|
|
* to zero) and corrupt data.
|
|
|
|
*/
|
|
|
|
if (page_ops && page_ops->iomap_valid) {
|
|
|
|
bool iomap_valid = page_ops->iomap_valid(iter->inode,
|
|
|
|
&iter->iomap);
|
|
|
|
if (!iomap_valid) {
|
|
|
|
iter->iomap.flags |= IOMAP_F_STALE;
|
|
|
|
status = 0;
|
|
|
|
goto out_unlock;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-10 04:47:44 +08:00
|
|
|
if (pos + len > folio_pos(folio) + folio_size(folio))
|
|
|
|
len = folio_pos(folio) + folio_size(folio) - pos;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2019-10-19 07:44:10 +08:00
|
|
|
if (srcmap->type == IOMAP_INLINE)
|
2021-05-02 23:33:08 +08:00
|
|
|
status = iomap_write_begin_inline(iter, folio);
|
2021-08-11 09:33:14 +08:00
|
|
|
else if (srcmap->flags & IOMAP_F_BUFFER_HEAD)
|
2021-11-04 02:05:47 +08:00
|
|
|
status = __block_write_begin_int(folio, pos, len, NULL, srcmap);
|
2019-07-15 23:50:59 +08:00
|
|
|
else
|
2021-05-02 23:33:08 +08:00
|
|
|
status = __iomap_write_begin(iter, pos, len, folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (unlikely(status))
|
|
|
|
goto out_unlock;
|
|
|
|
|
2021-05-02 23:33:08 +08:00
|
|
|
*foliop = folio;
|
2019-07-15 23:50:59 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_unlock:
|
2021-05-02 23:33:08 +08:00
|
|
|
folio_unlock(folio);
|
|
|
|
folio_put(folio);
|
2021-08-11 09:33:14 +08:00
|
|
|
iomap_write_failed(iter->inode, pos, len);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
out_no_page:
|
|
|
|
if (page_ops && page_ops->page_done)
|
2021-08-11 09:33:14 +08:00
|
|
|
page_ops->page_done(iter->inode, pos, 0, NULL);
|
2019-07-15 23:50:59 +08:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2020-09-21 23:58:41 +08:00
|
|
|
static size_t __iomap_write_end(struct inode *inode, loff_t pos, size_t len,
|
2021-05-02 23:33:08 +08:00
|
|
|
size_t copied, struct folio *folio)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-04-28 20:16:50 +08:00
|
|
|
struct iomap_page *iop = to_iomap_page(folio);
|
2021-05-02 23:33:08 +08:00
|
|
|
flush_dcache_folio(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The blocks that were entirely written will now be uptodate, so we
|
2022-04-29 20:54:32 +08:00
|
|
|
* don't have to worry about a read_folio reading them and overwriting a
|
2021-08-03 05:46:31 +08:00
|
|
|
* partial write. However, if we've encountered a short write and only
|
2019-07-15 23:50:59 +08:00
|
|
|
* partially written into a block, it will not be marked uptodate, so a
|
2022-04-29 20:54:32 +08:00
|
|
|
* read_folio might come in and destroy our partial write.
|
2019-07-15 23:50:59 +08:00
|
|
|
*
|
2021-08-03 05:46:31 +08:00
|
|
|
* Do the simplest thing and just treat any short write to a
|
|
|
|
* non-uptodate page as a zero-length write, and force the caller to
|
|
|
|
* redo the whole thing.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
2021-05-02 23:33:08 +08:00
|
|
|
if (unlikely(copied < len && !folio_test_uptodate(folio)))
|
2019-07-15 23:50:59 +08:00
|
|
|
return 0;
|
2021-04-28 20:20:48 +08:00
|
|
|
iomap_set_range_uptodate(folio, iop, offset_in_folio(folio, pos), len);
|
2021-05-02 23:33:08 +08:00
|
|
|
filemap_dirty_folio(inode->i_mapping, folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
return copied;
|
|
|
|
}
|
|
|
|
|
2021-08-11 09:33:16 +08:00
|
|
|
static size_t iomap_write_end_inline(const struct iomap_iter *iter,
|
2021-05-02 23:44:44 +08:00
|
|
|
struct folio *folio, loff_t pos, size_t copied)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *iomap = &iter->iomap;
|
2019-07-15 23:50:59 +08:00
|
|
|
void *addr;
|
|
|
|
|
2021-05-02 23:44:44 +08:00
|
|
|
WARN_ON_ONCE(!folio_test_uptodate(folio));
|
2021-08-04 00:38:22 +08:00
|
|
|
BUG_ON(!iomap_inline_data_valid(iomap));
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-05-02 23:44:44 +08:00
|
|
|
flush_dcache_folio(folio);
|
|
|
|
addr = kmap_local_folio(folio, pos);
|
2021-08-05 11:07:33 +08:00
|
|
|
memcpy(iomap_inline_data(iomap, pos), addr, copied);
|
|
|
|
kunmap_local(addr);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-08-11 09:33:14 +08:00
|
|
|
mark_inode_dirty(iter->inode);
|
2019-07-15 23:50:59 +08:00
|
|
|
return copied;
|
|
|
|
}
|
|
|
|
|
2020-09-21 23:58:41 +08:00
|
|
|
/* Returns the number of bytes copied. May be 0. Cannot be an errno. */
|
2021-08-11 09:33:14 +08:00
|
|
|
static size_t iomap_write_end(struct iomap_iter *iter, loff_t pos, size_t len,
|
2021-05-02 23:33:08 +08:00
|
|
|
size_t copied, struct folio *folio)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:14 +08:00
|
|
|
const struct iomap_page_ops *page_ops = iter->iomap.page_ops;
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *srcmap = iomap_iter_srcmap(iter);
|
2021-08-11 09:33:14 +08:00
|
|
|
loff_t old_size = iter->inode->i_size;
|
2020-09-21 23:58:41 +08:00
|
|
|
size_t ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2019-10-19 07:44:10 +08:00
|
|
|
if (srcmap->type == IOMAP_INLINE) {
|
2021-05-02 23:44:44 +08:00
|
|
|
ret = iomap_write_end_inline(iter, folio, pos, copied);
|
2019-10-19 07:44:10 +08:00
|
|
|
} else if (srcmap->flags & IOMAP_F_BUFFER_HEAD) {
|
2021-08-11 09:33:14 +08:00
|
|
|
ret = block_write_end(NULL, iter->inode->i_mapping, pos, len,
|
2021-05-02 23:33:08 +08:00
|
|
|
copied, &folio->page, NULL);
|
2019-07-15 23:50:59 +08:00
|
|
|
} else {
|
2021-05-02 23:33:08 +08:00
|
|
|
ret = __iomap_write_end(iter->inode, pos, len, copied, folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Update the in-memory inode size after copying the data into the page
|
|
|
|
* cache. It's up to the file system to write the updated size to disk,
|
|
|
|
* preferably after I/O completion so that no stale data is exposed.
|
|
|
|
*/
|
|
|
|
if (pos + ret > old_size) {
|
2021-08-11 09:33:14 +08:00
|
|
|
i_size_write(iter->inode, pos + ret);
|
|
|
|
iter->iomap.flags |= IOMAP_F_SIZE_CHANGED;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
2021-05-02 23:33:08 +08:00
|
|
|
folio_unlock(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (old_size < pos)
|
2021-08-11 09:33:14 +08:00
|
|
|
pagecache_isize_extended(iter->inode, old_size, pos);
|
2019-07-15 23:50:59 +08:00
|
|
|
if (page_ops && page_ops->page_done)
|
2021-05-02 23:33:08 +08:00
|
|
|
page_ops->page_done(iter->inode, pos, ret, &folio->page);
|
|
|
|
folio_put(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
if (ret < len)
|
2022-05-06 09:19:13 +08:00
|
|
|
iomap_write_failed(iter->inode, pos + ret, len - ret);
|
2019-07-15 23:50:59 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-08-11 09:33:08 +08:00
|
|
|
static loff_t iomap_write_iter(struct iomap_iter *iter, struct iov_iter *i)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:08 +08:00
|
|
|
loff_t length = iomap_length(iter);
|
|
|
|
loff_t pos = iter->pos;
|
2019-07-15 23:50:59 +08:00
|
|
|
ssize_t written = 0;
|
2021-08-11 09:33:08 +08:00
|
|
|
long status = 0;
|
2022-06-24 01:51:48 +08:00
|
|
|
struct address_space *mapping = iter->inode->i_mapping;
|
|
|
|
unsigned int bdp_flags = (iter->flags & IOMAP_NOWAIT) ? BDP_ASYNC : 0;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
do {
|
2021-05-02 23:33:08 +08:00
|
|
|
struct folio *folio;
|
2019-07-15 23:50:59 +08:00
|
|
|
struct page *page;
|
|
|
|
unsigned long offset; /* Offset into pagecache page */
|
|
|
|
unsigned long bytes; /* Bytes to write to page */
|
|
|
|
size_t copied; /* Bytes copied from user */
|
|
|
|
|
|
|
|
offset = offset_in_page(pos);
|
|
|
|
bytes = min_t(unsigned long, PAGE_SIZE - offset,
|
|
|
|
iov_iter_count(i));
|
|
|
|
again:
|
2022-06-24 01:51:48 +08:00
|
|
|
status = balance_dirty_pages_ratelimited_flags(mapping,
|
|
|
|
bdp_flags);
|
|
|
|
if (unlikely(status))
|
|
|
|
break;
|
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
if (bytes > length)
|
|
|
|
bytes = length;
|
|
|
|
|
|
|
|
/*
|
2021-08-03 05:46:31 +08:00
|
|
|
* Bring in the user page that we'll copy from _first_.
|
2019-07-15 23:50:59 +08:00
|
|
|
* Otherwise there's a nasty deadlock on copying from the
|
|
|
|
* same page as we're writing to, without it being marked
|
|
|
|
* up-to-date.
|
2022-06-24 01:51:48 +08:00
|
|
|
*
|
|
|
|
* For async buffered writes the assumption is that the user
|
|
|
|
* page has already been faulted in. This can be optimized by
|
|
|
|
* faulting the user page.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
2021-11-09 19:56:06 +08:00
|
|
|
if (unlikely(fault_in_iov_iter_readable(i, bytes) == bytes)) {
|
2019-07-15 23:50:59 +08:00
|
|
|
status = -EFAULT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-05-02 23:33:08 +08:00
|
|
|
status = iomap_write_begin(iter, pos, bytes, &folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
if (unlikely(status))
|
|
|
|
break;
|
2022-11-29 06:09:17 +08:00
|
|
|
if (iter->iomap.flags & IOMAP_F_STALE)
|
|
|
|
break;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-05-02 23:33:08 +08:00
|
|
|
page = folio_file_page(folio, pos >> PAGE_SHIFT);
|
2022-06-24 01:51:48 +08:00
|
|
|
if (mapping_writably_mapped(mapping))
|
2019-07-15 23:50:59 +08:00
|
|
|
flush_dcache_page(page);
|
|
|
|
|
2021-04-30 22:26:41 +08:00
|
|
|
copied = copy_page_from_iter_atomic(page, offset, bytes, i);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-05-02 23:33:08 +08:00
|
|
|
status = iomap_write_end(iter, pos, bytes, copied, folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-04-30 22:26:41 +08:00
|
|
|
if (unlikely(copied != status))
|
|
|
|
iov_iter_revert(i, copied - status);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-04-30 22:26:41 +08:00
|
|
|
cond_resched();
|
2021-05-31 12:32:44 +08:00
|
|
|
if (unlikely(status == 0)) {
|
2019-07-15 23:50:59 +08:00
|
|
|
/*
|
2021-05-31 12:32:44 +08:00
|
|
|
* A short copy made iomap_write_end() reject the
|
|
|
|
* thing entirely. Might be memory poisoning
|
|
|
|
* halfway through, might be a race with munmap,
|
|
|
|
* might be severe memory pressure.
|
2019-07-15 23:50:59 +08:00
|
|
|
*/
|
2021-05-31 12:32:44 +08:00
|
|
|
if (copied)
|
|
|
|
bytes = copied;
|
2019-07-15 23:50:59 +08:00
|
|
|
goto again;
|
|
|
|
}
|
2021-04-30 22:26:41 +08:00
|
|
|
pos += status;
|
|
|
|
written += status;
|
|
|
|
length -= status;
|
2019-07-15 23:50:59 +08:00
|
|
|
} while (iov_iter_count(i) && length);
|
|
|
|
|
2022-06-24 01:51:49 +08:00
|
|
|
if (status == -EAGAIN) {
|
|
|
|
iov_iter_revert(i, written);
|
|
|
|
return -EAGAIN;
|
|
|
|
}
|
2019-07-15 23:50:59 +08:00
|
|
|
return written ? written : status;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t
|
2021-08-11 09:33:08 +08:00
|
|
|
iomap_file_buffered_write(struct kiocb *iocb, struct iov_iter *i,
|
2019-07-15 23:50:59 +08:00
|
|
|
const struct iomap_ops *ops)
|
|
|
|
{
|
2021-08-11 09:33:08 +08:00
|
|
|
struct iomap_iter iter = {
|
|
|
|
.inode = iocb->ki_filp->f_mapping->host,
|
|
|
|
.pos = iocb->ki_pos,
|
|
|
|
.len = iov_iter_count(i),
|
|
|
|
.flags = IOMAP_WRITE,
|
|
|
|
};
|
|
|
|
int ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2022-06-24 01:51:48 +08:00
|
|
|
if (iocb->ki_flags & IOCB_NOWAIT)
|
|
|
|
iter.flags |= IOMAP_NOWAIT;
|
|
|
|
|
2021-08-11 09:33:08 +08:00
|
|
|
while ((ret = iomap_iter(&iter, ops)) > 0)
|
|
|
|
iter.processed = iomap_write_iter(&iter, i);
|
|
|
|
if (iter.pos == iocb->ki_pos)
|
|
|
|
return ret;
|
|
|
|
return iter.pos - iocb->ki_pos;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_file_buffered_write);
|
|
|
|
|
iomap: buffered write failure should not truncate the page cache
iomap_file_buffered_write_punch_delalloc() currently invalidates the
page cache over the unused range of the delalloc extent that was
allocated. While the write allocated the delalloc extent, it does
not own it exclusively as the write does not hold any locks that
prevent either writeback or mmap page faults from changing the state
of either the page cache or the extent state backing this range.
Whilst xfs_bmap_punch_delalloc_range() already handles races in
extent conversion - it will only punch out delalloc extents and it
ignores any other type of extent - the page cache truncate does not
discriminate between data written by this write or some other task.
As a result, truncating the page cache can result in data corruption
if the write races with mmap modifications to the file over the same
range.
generic/346 exercises this workload, and if we randomly fail writes
(as will happen when iomap gets stale iomap detection later in the
patchset), it will randomly corrupt the file data because it removes
data written by mmap() in the same page as the write() that failed.
Hence we do not want to punch out the page cache over the range of
the extent we failed to write to - what we actually need to do is
detect the ranges that have dirty data in cache over them and *not
punch them out*.
To do this, we have to walk the page cache over the range of the
delalloc extent we want to remove. This is made complex by the fact
we have to handle partially up-to-date folios correctly and this can
happen even when the FSB size == PAGE_SIZE because we now support
multi-page folios in the page cache.
Because we are only interested in discovering the edges of data
ranges in the page cache (i.e. hole-data boundaries) we can make use
of mapping_seek_hole_data() to find those transitions in the page
cache. As we hold the invalidate_lock, we know that the boundaries
are not going to change while we walk the range. This interface is
also byte-based and is sub-page block aware, so we can find the data
ranges in the cache based on byte offsets rather than page, folio or
fs block sized chunks. This greatly simplifies the logic of finding
dirty cached ranges in the page cache.
Once we've identified a range that contains cached data, we can then
iterate the range folio by folio. This allows us to determine if the
data is dirty and hence perform the correct delalloc extent punching
operations. The seek interface we use to iterate data ranges will
give us sub-folio start/end granularity, so we may end up looking up
the same folio multiple times as the seek interface iterates across
each discontiguous data region in the folio.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
2022-11-29 06:09:11 +08:00
|
|
|
/*
|
|
|
|
* Scan the data range passed to us for dirty page cache folios. If we find a
|
|
|
|
* dirty folio, punch out the preceeding range and update the offset from which
|
|
|
|
* the next punch will start from.
|
|
|
|
*
|
|
|
|
* We can punch out storage reservations under clean pages because they either
|
|
|
|
* contain data that has been written back - in which case the delalloc punch
|
|
|
|
* over that range is a no-op - or they have been read faults in which case they
|
|
|
|
* contain zeroes and we can remove the delalloc backing range and any new
|
|
|
|
* writes to those pages will do the normal hole filling operation...
|
|
|
|
*
|
|
|
|
* This makes the logic simple: we only need to keep the delalloc extents only
|
|
|
|
* over the dirty ranges of the page cache.
|
|
|
|
*
|
|
|
|
* This function uses [start_byte, end_byte) intervals (i.e. open ended) to
|
|
|
|
* simplify range iterations.
|
|
|
|
*/
|
|
|
|
static int iomap_write_delalloc_scan(struct inode *inode,
|
|
|
|
loff_t *punch_start_byte, loff_t start_byte, loff_t end_byte,
|
|
|
|
int (*punch)(struct inode *inode, loff_t offset, loff_t length))
|
|
|
|
{
|
|
|
|
while (start_byte < end_byte) {
|
|
|
|
struct folio *folio;
|
|
|
|
|
|
|
|
/* grab locked page */
|
|
|
|
folio = filemap_lock_folio(inode->i_mapping,
|
|
|
|
start_byte >> PAGE_SHIFT);
|
|
|
|
if (!folio) {
|
|
|
|
start_byte = ALIGN_DOWN(start_byte, PAGE_SIZE) +
|
|
|
|
PAGE_SIZE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if dirty, punch up to offset */
|
|
|
|
if (folio_test_dirty(folio)) {
|
|
|
|
if (start_byte > *punch_start_byte) {
|
|
|
|
int error;
|
|
|
|
|
|
|
|
error = punch(inode, *punch_start_byte,
|
|
|
|
start_byte - *punch_start_byte);
|
|
|
|
if (error) {
|
|
|
|
folio_unlock(folio);
|
|
|
|
folio_put(folio);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure the next punch start is correctly bound to
|
|
|
|
* the end of this data range, not the end of the folio.
|
|
|
|
*/
|
|
|
|
*punch_start_byte = min_t(loff_t, end_byte,
|
|
|
|
folio_next_index(folio) << PAGE_SHIFT);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* move offset to start of next folio in range */
|
|
|
|
start_byte = folio_next_index(folio) << PAGE_SHIFT;
|
|
|
|
folio_unlock(folio);
|
|
|
|
folio_put(folio);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Punch out all the delalloc blocks in the range given except for those that
|
|
|
|
* have dirty data still pending in the page cache - those are going to be
|
|
|
|
* written and so must still retain the delalloc backing for writeback.
|
|
|
|
*
|
|
|
|
* As we are scanning the page cache for data, we don't need to reimplement the
|
|
|
|
* wheel - mapping_seek_hole_data() does exactly what we need to identify the
|
|
|
|
* start and end of data ranges correctly even for sub-folio block sizes. This
|
|
|
|
* byte range based iteration is especially convenient because it means we
|
|
|
|
* don't have to care about variable size folios, nor where the start or end of
|
|
|
|
* the data range lies within a folio, if they lie within the same folio or even
|
|
|
|
* if there are multiple discontiguous data ranges within the folio.
|
|
|
|
*
|
|
|
|
* It should be noted that mapping_seek_hole_data() is not aware of EOF, and so
|
|
|
|
* can return data ranges that exist in the cache beyond EOF. e.g. a page fault
|
|
|
|
* spanning EOF will initialise the post-EOF data to zeroes and mark it up to
|
|
|
|
* date. A write page fault can then mark it dirty. If we then fail a write()
|
|
|
|
* beyond EOF into that up to date cached range, we allocate a delalloc block
|
|
|
|
* beyond EOF and then have to punch it out. Because the range is up to date,
|
|
|
|
* mapping_seek_hole_data() will return it, and we will skip the punch because
|
|
|
|
* the folio is dirty. THis is incorrect - we always need to punch out delalloc
|
|
|
|
* beyond EOF in this case as writeback will never write back and covert that
|
|
|
|
* delalloc block beyond EOF. Hence we limit the cached data scan range to EOF,
|
|
|
|
* resulting in always punching out the range from the EOF to the end of the
|
|
|
|
* range the iomap spans.
|
|
|
|
*
|
|
|
|
* Intervals are of the form [start_byte, end_byte) (i.e. open ended) because it
|
|
|
|
* matches the intervals returned by mapping_seek_hole_data(). i.e. SEEK_DATA
|
|
|
|
* returns the start of a data range (start_byte), and SEEK_HOLE(start_byte)
|
|
|
|
* returns the end of the data range (data_end). Using closed intervals would
|
|
|
|
* require sprinkling this code with magic "+ 1" and "- 1" arithmetic and expose
|
|
|
|
* the code to subtle off-by-one bugs....
|
|
|
|
*/
|
|
|
|
static int iomap_write_delalloc_release(struct inode *inode,
|
|
|
|
loff_t start_byte, loff_t end_byte,
|
|
|
|
int (*punch)(struct inode *inode, loff_t pos, loff_t length))
|
|
|
|
{
|
|
|
|
loff_t punch_start_byte = start_byte;
|
|
|
|
loff_t scan_end_byte = min(i_size_read(inode), end_byte);
|
|
|
|
int error = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Lock the mapping to avoid races with page faults re-instantiating
|
|
|
|
* folios and dirtying them via ->page_mkwrite whilst we walk the
|
|
|
|
* cache and perform delalloc extent removal. Failing to do this can
|
|
|
|
* leave dirty pages with no space reservation in the cache.
|
|
|
|
*/
|
|
|
|
filemap_invalidate_lock(inode->i_mapping);
|
|
|
|
while (start_byte < scan_end_byte) {
|
|
|
|
loff_t data_end;
|
|
|
|
|
|
|
|
start_byte = mapping_seek_hole_data(inode->i_mapping,
|
|
|
|
start_byte, scan_end_byte, SEEK_DATA);
|
|
|
|
/*
|
|
|
|
* If there is no more data to scan, all that is left is to
|
|
|
|
* punch out the remaining range.
|
|
|
|
*/
|
|
|
|
if (start_byte == -ENXIO || start_byte == scan_end_byte)
|
|
|
|
break;
|
|
|
|
if (start_byte < 0) {
|
|
|
|
error = start_byte;
|
|
|
|
goto out_unlock;
|
|
|
|
}
|
|
|
|
WARN_ON_ONCE(start_byte < punch_start_byte);
|
|
|
|
WARN_ON_ONCE(start_byte > scan_end_byte);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We find the end of this contiguous cached data range by
|
|
|
|
* seeking from start_byte to the beginning of the next hole.
|
|
|
|
*/
|
|
|
|
data_end = mapping_seek_hole_data(inode->i_mapping, start_byte,
|
|
|
|
scan_end_byte, SEEK_HOLE);
|
|
|
|
if (data_end < 0) {
|
|
|
|
error = data_end;
|
|
|
|
goto out_unlock;
|
|
|
|
}
|
|
|
|
WARN_ON_ONCE(data_end <= start_byte);
|
|
|
|
WARN_ON_ONCE(data_end > scan_end_byte);
|
|
|
|
|
|
|
|
error = iomap_write_delalloc_scan(inode, &punch_start_byte,
|
|
|
|
start_byte, data_end, punch);
|
|
|
|
if (error)
|
|
|
|
goto out_unlock;
|
|
|
|
|
|
|
|
/* The next data search starts at the end of this one. */
|
|
|
|
start_byte = data_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (punch_start_byte < end_byte)
|
|
|
|
error = punch(inode, punch_start_byte,
|
|
|
|
end_byte - punch_start_byte);
|
|
|
|
out_unlock:
|
|
|
|
filemap_invalidate_unlock(inode->i_mapping);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2022-11-23 09:44:38 +08:00
|
|
|
/*
|
|
|
|
* When a short write occurs, the filesystem may need to remove reserved space
|
|
|
|
* that was allocated in ->iomap_begin from it's ->iomap_end method. For
|
|
|
|
* filesystems that use delayed allocation, we need to punch out delalloc
|
|
|
|
* extents from the range that are not dirty in the page cache. As the write can
|
|
|
|
* race with page faults, there can be dirty pages over the delalloc extent
|
|
|
|
* outside the range of a short write but still within the delalloc extent
|
|
|
|
* allocated for this iomap.
|
|
|
|
*
|
|
|
|
* This function uses [start_byte, end_byte) intervals (i.e. open ended) to
|
iomap: buffered write failure should not truncate the page cache
iomap_file_buffered_write_punch_delalloc() currently invalidates the
page cache over the unused range of the delalloc extent that was
allocated. While the write allocated the delalloc extent, it does
not own it exclusively as the write does not hold any locks that
prevent either writeback or mmap page faults from changing the state
of either the page cache or the extent state backing this range.
Whilst xfs_bmap_punch_delalloc_range() already handles races in
extent conversion - it will only punch out delalloc extents and it
ignores any other type of extent - the page cache truncate does not
discriminate between data written by this write or some other task.
As a result, truncating the page cache can result in data corruption
if the write races with mmap modifications to the file over the same
range.
generic/346 exercises this workload, and if we randomly fail writes
(as will happen when iomap gets stale iomap detection later in the
patchset), it will randomly corrupt the file data because it removes
data written by mmap() in the same page as the write() that failed.
Hence we do not want to punch out the page cache over the range of
the extent we failed to write to - what we actually need to do is
detect the ranges that have dirty data in cache over them and *not
punch them out*.
To do this, we have to walk the page cache over the range of the
delalloc extent we want to remove. This is made complex by the fact
we have to handle partially up-to-date folios correctly and this can
happen even when the FSB size == PAGE_SIZE because we now support
multi-page folios in the page cache.
Because we are only interested in discovering the edges of data
ranges in the page cache (i.e. hole-data boundaries) we can make use
of mapping_seek_hole_data() to find those transitions in the page
cache. As we hold the invalidate_lock, we know that the boundaries
are not going to change while we walk the range. This interface is
also byte-based and is sub-page block aware, so we can find the data
ranges in the cache based on byte offsets rather than page, folio or
fs block sized chunks. This greatly simplifies the logic of finding
dirty cached ranges in the page cache.
Once we've identified a range that contains cached data, we can then
iterate the range folio by folio. This allows us to determine if the
data is dirty and hence perform the correct delalloc extent punching
operations. The seek interface we use to iterate data ranges will
give us sub-folio start/end granularity, so we may end up looking up
the same folio multiple times as the seek interface iterates across
each discontiguous data region in the folio.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
2022-11-29 06:09:11 +08:00
|
|
|
* simplify range iterations.
|
|
|
|
*
|
|
|
|
* The punch() callback *must* only punch delalloc extents in the range passed
|
|
|
|
* to it. It must skip over all other types of extents in the range and leave
|
|
|
|
* them completely unchanged. It must do this punch atomically with respect to
|
|
|
|
* other extent modifications.
|
|
|
|
*
|
|
|
|
* The punch() callback may be called with a folio locked to prevent writeback
|
|
|
|
* extent allocation racing at the edge of the range we are currently punching.
|
|
|
|
* The locked folio may or may not cover the range being punched, so it is not
|
|
|
|
* safe for the punch() callback to lock folios itself.
|
|
|
|
*
|
|
|
|
* Lock order is:
|
|
|
|
*
|
|
|
|
* inode->i_rwsem (shared or exclusive)
|
|
|
|
* inode->i_mapping->invalidate_lock (exclusive)
|
|
|
|
* folio_lock()
|
|
|
|
* ->punch
|
|
|
|
* internal filesystem allocation lock
|
2022-11-23 09:44:38 +08:00
|
|
|
*/
|
|
|
|
int iomap_file_buffered_write_punch_delalloc(struct inode *inode,
|
|
|
|
struct iomap *iomap, loff_t pos, loff_t length,
|
|
|
|
ssize_t written,
|
|
|
|
int (*punch)(struct inode *inode, loff_t pos, loff_t length))
|
|
|
|
{
|
|
|
|
loff_t start_byte;
|
|
|
|
loff_t end_byte;
|
|
|
|
int blocksize = i_blocksize(inode);
|
|
|
|
|
|
|
|
if (iomap->type != IOMAP_DELALLOC)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* If we didn't reserve the blocks, we're not allowed to punch them. */
|
|
|
|
if (!(iomap->flags & IOMAP_F_NEW))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* start_byte refers to the first unused block after a short write. If
|
|
|
|
* nothing was written, round offset down to point at the first block in
|
|
|
|
* the range.
|
|
|
|
*/
|
|
|
|
if (unlikely(!written))
|
|
|
|
start_byte = round_down(pos, blocksize);
|
|
|
|
else
|
|
|
|
start_byte = round_up(pos + written, blocksize);
|
|
|
|
end_byte = round_up(pos + length, blocksize);
|
|
|
|
|
|
|
|
/* Nothing to do if we've written the entire delalloc extent */
|
|
|
|
if (start_byte >= end_byte)
|
|
|
|
return 0;
|
|
|
|
|
iomap: buffered write failure should not truncate the page cache
iomap_file_buffered_write_punch_delalloc() currently invalidates the
page cache over the unused range of the delalloc extent that was
allocated. While the write allocated the delalloc extent, it does
not own it exclusively as the write does not hold any locks that
prevent either writeback or mmap page faults from changing the state
of either the page cache or the extent state backing this range.
Whilst xfs_bmap_punch_delalloc_range() already handles races in
extent conversion - it will only punch out delalloc extents and it
ignores any other type of extent - the page cache truncate does not
discriminate between data written by this write or some other task.
As a result, truncating the page cache can result in data corruption
if the write races with mmap modifications to the file over the same
range.
generic/346 exercises this workload, and if we randomly fail writes
(as will happen when iomap gets stale iomap detection later in the
patchset), it will randomly corrupt the file data because it removes
data written by mmap() in the same page as the write() that failed.
Hence we do not want to punch out the page cache over the range of
the extent we failed to write to - what we actually need to do is
detect the ranges that have dirty data in cache over them and *not
punch them out*.
To do this, we have to walk the page cache over the range of the
delalloc extent we want to remove. This is made complex by the fact
we have to handle partially up-to-date folios correctly and this can
happen even when the FSB size == PAGE_SIZE because we now support
multi-page folios in the page cache.
Because we are only interested in discovering the edges of data
ranges in the page cache (i.e. hole-data boundaries) we can make use
of mapping_seek_hole_data() to find those transitions in the page
cache. As we hold the invalidate_lock, we know that the boundaries
are not going to change while we walk the range. This interface is
also byte-based and is sub-page block aware, so we can find the data
ranges in the cache based on byte offsets rather than page, folio or
fs block sized chunks. This greatly simplifies the logic of finding
dirty cached ranges in the page cache.
Once we've identified a range that contains cached data, we can then
iterate the range folio by folio. This allows us to determine if the
data is dirty and hence perform the correct delalloc extent punching
operations. The seek interface we use to iterate data ranges will
give us sub-folio start/end granularity, so we may end up looking up
the same folio multiple times as the seek interface iterates across
each discontiguous data region in the folio.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
2022-11-29 06:09:11 +08:00
|
|
|
return iomap_write_delalloc_release(inode, start_byte, end_byte,
|
|
|
|
punch);
|
2022-11-23 09:44:38 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_file_buffered_write_punch_delalloc);
|
|
|
|
|
2021-08-11 09:33:09 +08:00
|
|
|
static loff_t iomap_unshare_iter(struct iomap_iter *iter)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:09 +08:00
|
|
|
struct iomap *iomap = &iter->iomap;
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *srcmap = iomap_iter_srcmap(iter);
|
2021-08-11 09:33:09 +08:00
|
|
|
loff_t pos = iter->pos;
|
|
|
|
loff_t length = iomap_length(iter);
|
2019-07-15 23:50:59 +08:00
|
|
|
long status = 0;
|
2020-06-09 11:58:29 +08:00
|
|
|
loff_t written = 0;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2019-10-19 07:41:34 +08:00
|
|
|
/* don't bother with blocks that are not shared to start with */
|
|
|
|
if (!(iomap->flags & IOMAP_F_SHARED))
|
|
|
|
return length;
|
|
|
|
/* don't bother with holes or unwritten extents */
|
2019-10-19 07:44:10 +08:00
|
|
|
if (srcmap->type == IOMAP_HOLE || srcmap->type == IOMAP_UNWRITTEN)
|
2019-10-19 07:41:34 +08:00
|
|
|
return length;
|
|
|
|
|
2019-07-15 23:50:59 +08:00
|
|
|
do {
|
2019-10-19 07:42:50 +08:00
|
|
|
unsigned long offset = offset_in_page(pos);
|
|
|
|
unsigned long bytes = min_t(loff_t, PAGE_SIZE - offset, length);
|
2021-05-02 23:33:08 +08:00
|
|
|
struct folio *folio;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-05-02 23:33:08 +08:00
|
|
|
status = iomap_write_begin(iter, pos, bytes, &folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
if (unlikely(status))
|
|
|
|
return status;
|
2022-11-29 06:09:17 +08:00
|
|
|
if (iter->iomap.flags & IOMAP_F_STALE)
|
|
|
|
break;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-05-02 23:33:08 +08:00
|
|
|
status = iomap_write_end(iter, pos, bytes, bytes, folio);
|
2020-09-21 23:58:41 +08:00
|
|
|
if (WARN_ON_ONCE(status == 0))
|
|
|
|
return -EIO;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
cond_resched();
|
|
|
|
|
|
|
|
pos += status;
|
|
|
|
written += status;
|
|
|
|
length -= status;
|
|
|
|
|
2021-08-11 09:33:09 +08:00
|
|
|
balance_dirty_pages_ratelimited(iter->inode->i_mapping);
|
2019-07-15 23:50:59 +08:00
|
|
|
} while (length);
|
|
|
|
|
|
|
|
return written;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2019-10-19 07:41:34 +08:00
|
|
|
iomap_file_unshare(struct inode *inode, loff_t pos, loff_t len,
|
2019-07-15 23:50:59 +08:00
|
|
|
const struct iomap_ops *ops)
|
|
|
|
{
|
2021-08-11 09:33:09 +08:00
|
|
|
struct iomap_iter iter = {
|
|
|
|
.inode = inode,
|
|
|
|
.pos = pos,
|
|
|
|
.len = len,
|
2021-08-11 09:33:14 +08:00
|
|
|
.flags = IOMAP_WRITE | IOMAP_UNSHARE,
|
2021-08-11 09:33:09 +08:00
|
|
|
};
|
|
|
|
int ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-08-11 09:33:09 +08:00
|
|
|
while ((ret = iomap_iter(&iter, ops)) > 0)
|
|
|
|
iter.processed = iomap_unshare_iter(&iter);
|
|
|
|
return ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
2019-10-19 07:41:34 +08:00
|
|
|
EXPORT_SYMBOL_GPL(iomap_file_unshare);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-08-11 09:33:09 +08:00
|
|
|
static loff_t iomap_zero_iter(struct iomap_iter *iter, bool *did_zero)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:16 +08:00
|
|
|
const struct iomap *srcmap = iomap_iter_srcmap(iter);
|
2021-08-11 09:33:09 +08:00
|
|
|
loff_t pos = iter->pos;
|
|
|
|
loff_t length = iomap_length(iter);
|
2019-07-15 23:50:59 +08:00
|
|
|
loff_t written = 0;
|
|
|
|
|
|
|
|
/* already zeroed? we're done. */
|
2019-10-19 07:44:10 +08:00
|
|
|
if (srcmap->type == IOMAP_HOLE || srcmap->type == IOMAP_UNWRITTEN)
|
2020-09-21 23:58:42 +08:00
|
|
|
return length;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
do {
|
2021-12-21 08:03:46 +08:00
|
|
|
struct folio *folio;
|
|
|
|
int status;
|
|
|
|
size_t offset;
|
|
|
|
size_t bytes = min_t(u64, SIZE_MAX, length);
|
|
|
|
|
|
|
|
status = iomap_write_begin(iter, pos, bytes, &folio);
|
|
|
|
if (status)
|
|
|
|
return status;
|
2022-11-29 06:09:17 +08:00
|
|
|
if (iter->iomap.flags & IOMAP_F_STALE)
|
|
|
|
break;
|
2021-12-21 08:03:46 +08:00
|
|
|
|
|
|
|
offset = offset_in_folio(folio, pos);
|
|
|
|
if (bytes > folio_size(folio) - offset)
|
|
|
|
bytes = folio_size(folio) - offset;
|
|
|
|
|
|
|
|
folio_zero_range(folio, offset, bytes);
|
|
|
|
folio_mark_accessed(folio);
|
|
|
|
|
|
|
|
bytes = iomap_write_end(iter, pos, bytes, bytes, folio);
|
|
|
|
if (WARN_ON_ONCE(bytes == 0))
|
|
|
|
return -EIO;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
|
|
|
pos += bytes;
|
2020-09-21 23:58:42 +08:00
|
|
|
length -= bytes;
|
2019-07-15 23:50:59 +08:00
|
|
|
written += bytes;
|
2020-09-21 23:58:42 +08:00
|
|
|
} while (length > 0);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2022-07-01 01:04:18 +08:00
|
|
|
if (did_zero)
|
|
|
|
*did_zero = true;
|
2019-07-15 23:50:59 +08:00
|
|
|
return written;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
iomap_zero_range(struct inode *inode, loff_t pos, loff_t len, bool *did_zero,
|
|
|
|
const struct iomap_ops *ops)
|
|
|
|
{
|
2021-08-11 09:33:09 +08:00
|
|
|
struct iomap_iter iter = {
|
|
|
|
.inode = inode,
|
|
|
|
.pos = pos,
|
|
|
|
.len = len,
|
|
|
|
.flags = IOMAP_ZERO,
|
|
|
|
};
|
|
|
|
int ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-08-11 09:33:09 +08:00
|
|
|
while ((ret = iomap_iter(&iter, ops)) > 0)
|
|
|
|
iter.processed = iomap_zero_iter(&iter, did_zero);
|
|
|
|
return ret;
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_zero_range);
|
|
|
|
|
|
|
|
int
|
|
|
|
iomap_truncate_page(struct inode *inode, loff_t pos, bool *did_zero,
|
|
|
|
const struct iomap_ops *ops)
|
|
|
|
{
|
|
|
|
unsigned int blocksize = i_blocksize(inode);
|
|
|
|
unsigned int off = pos & (blocksize - 1);
|
|
|
|
|
|
|
|
/* Block boundary? Nothing to do */
|
|
|
|
if (!off)
|
|
|
|
return 0;
|
|
|
|
return iomap_zero_range(inode, pos, blocksize - off, did_zero, ops);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_truncate_page);
|
|
|
|
|
2021-04-29 10:32:02 +08:00
|
|
|
static loff_t iomap_folio_mkwrite_iter(struct iomap_iter *iter,
|
|
|
|
struct folio *folio)
|
2019-07-15 23:50:59 +08:00
|
|
|
{
|
2021-08-11 09:33:09 +08:00
|
|
|
loff_t length = iomap_length(iter);
|
2019-07-15 23:50:59 +08:00
|
|
|
int ret;
|
|
|
|
|
2021-08-11 09:33:09 +08:00
|
|
|
if (iter->iomap.flags & IOMAP_F_BUFFER_HEAD) {
|
2021-11-04 02:05:47 +08:00
|
|
|
ret = __block_write_begin_int(folio, iter->pos, length, NULL,
|
2021-08-11 09:33:09 +08:00
|
|
|
&iter->iomap);
|
2019-07-15 23:50:59 +08:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
2021-04-29 10:32:02 +08:00
|
|
|
block_commit_write(&folio->page, 0, length);
|
2019-07-15 23:50:59 +08:00
|
|
|
} else {
|
2021-04-29 10:32:02 +08:00
|
|
|
WARN_ON_ONCE(!folio_test_uptodate(folio));
|
|
|
|
folio_mark_dirty(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
vm_fault_t iomap_page_mkwrite(struct vm_fault *vmf, const struct iomap_ops *ops)
|
|
|
|
{
|
2021-08-11 09:33:09 +08:00
|
|
|
struct iomap_iter iter = {
|
|
|
|
.inode = file_inode(vmf->vma->vm_file),
|
|
|
|
.flags = IOMAP_WRITE | IOMAP_FAULT,
|
|
|
|
};
|
2021-04-29 10:32:02 +08:00
|
|
|
struct folio *folio = page_folio(vmf->page);
|
2019-07-15 23:50:59 +08:00
|
|
|
ssize_t ret;
|
|
|
|
|
2021-04-29 10:32:02 +08:00
|
|
|
folio_lock(folio);
|
|
|
|
ret = folio_mkwrite_check_truncate(folio, iter.inode);
|
2020-01-07 00:58:23 +08:00
|
|
|
if (ret < 0)
|
2019-07-15 23:50:59 +08:00
|
|
|
goto out_unlock;
|
2021-04-29 10:32:02 +08:00
|
|
|
iter.pos = folio_pos(folio);
|
2021-08-11 09:33:09 +08:00
|
|
|
iter.len = ret;
|
|
|
|
while ((ret = iomap_iter(&iter, ops)) > 0)
|
2021-04-29 10:32:02 +08:00
|
|
|
iter.processed = iomap_folio_mkwrite_iter(&iter, folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
|
2021-08-11 09:33:09 +08:00
|
|
|
if (ret < 0)
|
|
|
|
goto out_unlock;
|
2021-04-29 10:32:02 +08:00
|
|
|
folio_wait_stable(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
return VM_FAULT_LOCKED;
|
|
|
|
out_unlock:
|
2021-04-29 10:32:02 +08:00
|
|
|
folio_unlock(folio);
|
2019-07-15 23:50:59 +08:00
|
|
|
return block_page_mkwrite_return(ret);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_page_mkwrite);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
2021-01-02 05:53:26 +08:00
|
|
|
static void iomap_finish_folio_write(struct inode *inode, struct folio *folio,
|
|
|
|
size_t len, int error)
|
2019-10-18 04:12:15 +08:00
|
|
|
{
|
2021-04-28 10:56:17 +08:00
|
|
|
struct iomap_page *iop = to_iomap_page(folio);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
if (error) {
|
2021-01-02 05:53:26 +08:00
|
|
|
folio_set_error(folio);
|
2021-08-11 09:32:55 +08:00
|
|
|
mapping_set_error(inode->i_mapping, error);
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
|
2021-01-02 05:53:26 +08:00
|
|
|
WARN_ON_ONCE(i_blocks_per_folio(inode, folio) > 1 && !iop);
|
2020-09-21 23:58:41 +08:00
|
|
|
WARN_ON_ONCE(iop && atomic_read(&iop->write_bytes_pending) <= 0);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
2020-09-21 23:58:41 +08:00
|
|
|
if (!iop || atomic_sub_and_test(len, &iop->write_bytes_pending))
|
2021-01-02 05:53:26 +08:00
|
|
|
folio_end_writeback(folio);
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We're now finished for good with this ioend structure. Update the page
|
|
|
|
* state, release holds on bios, and finally free up memory. Do not use the
|
|
|
|
* ioend after this.
|
|
|
|
*/
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
static u32
|
2019-10-18 04:12:15 +08:00
|
|
|
iomap_finish_ioend(struct iomap_ioend *ioend, int error)
|
|
|
|
{
|
|
|
|
struct inode *inode = ioend->io_inode;
|
|
|
|
struct bio *bio = &ioend->io_inline_bio;
|
|
|
|
struct bio *last = ioend->io_bio, *next;
|
|
|
|
u64 start = bio->bi_iter.bi_sector;
|
2019-12-05 14:59:02 +08:00
|
|
|
loff_t offset = ioend->io_offset;
|
2019-10-18 04:12:15 +08:00
|
|
|
bool quiet = bio_flagged(bio, BIO_QUIET);
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
u32 folio_count = 0;
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
for (bio = &ioend->io_inline_bio; bio; bio = next) {
|
2021-01-02 05:53:26 +08:00
|
|
|
struct folio_iter fi;
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* For the last bio, bi_private points to the ioend, so we
|
|
|
|
* need to explicitly end the iteration here.
|
|
|
|
*/
|
|
|
|
if (bio == last)
|
|
|
|
next = NULL;
|
|
|
|
else
|
|
|
|
next = bio->bi_private;
|
|
|
|
|
2021-01-02 05:53:26 +08:00
|
|
|
/* walk all folios in bio, ending page IO on them */
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
bio_for_each_folio_all(fi, bio) {
|
2021-01-02 05:53:26 +08:00
|
|
|
iomap_finish_folio_write(inode, fi.folio, fi.length,
|
|
|
|
error);
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
folio_count++;
|
|
|
|
}
|
2019-10-18 04:12:15 +08:00
|
|
|
bio_put(bio);
|
|
|
|
}
|
2019-12-05 14:59:02 +08:00
|
|
|
/* The ioend has been freed by bio_put() */
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
if (unlikely(error && !quiet)) {
|
|
|
|
printk_ratelimited(KERN_ERR
|
2019-10-18 05:02:07 +08:00
|
|
|
"%s: writeback error on inode %lu, offset %lld, sector %llu",
|
2019-12-05 14:59:02 +08:00
|
|
|
inode->i_sb->s_id, inode->i_ino, offset, start);
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
return folio_count;
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
/*
|
|
|
|
* Ioend completion routine for merged bios. This can only be called from task
|
|
|
|
* contexts as merged ioends can be of unbound length. Hence we have to break up
|
|
|
|
* the writeback completions into manageable chunks to avoid long scheduler
|
|
|
|
* holdoffs. We aim to keep scheduler holdoffs down below 10ms so that we get
|
|
|
|
* good batch processing throughput without creating adverse scheduler latency
|
|
|
|
* conditions.
|
|
|
|
*/
|
2019-10-18 04:12:15 +08:00
|
|
|
void
|
|
|
|
iomap_finish_ioends(struct iomap_ioend *ioend, int error)
|
|
|
|
{
|
|
|
|
struct list_head tmp;
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
u32 completions;
|
|
|
|
|
|
|
|
might_sleep();
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
list_replace_init(&ioend->io_list, &tmp);
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
completions = iomap_finish_ioend(ioend, error);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
while (!list_empty(&tmp)) {
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
if (completions > IOEND_BATCH_SIZE * 8) {
|
|
|
|
cond_resched();
|
|
|
|
completions = 0;
|
|
|
|
}
|
2019-10-18 04:12:15 +08:00
|
|
|
ioend = list_first_entry(&tmp, struct iomap_ioend, io_list);
|
|
|
|
list_del_init(&ioend->io_list);
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
completions += iomap_finish_ioend(ioend, error);
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_finish_ioends);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We can merge two adjacent ioends if they have the same set of work to do.
|
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
iomap_ioend_can_merge(struct iomap_ioend *ioend, struct iomap_ioend *next)
|
|
|
|
{
|
|
|
|
if (ioend->io_bio->bi_status != next->io_bio->bi_status)
|
|
|
|
return false;
|
|
|
|
if ((ioend->io_flags & IOMAP_F_SHARED) ^
|
|
|
|
(next->io_flags & IOMAP_F_SHARED))
|
|
|
|
return false;
|
|
|
|
if ((ioend->io_type == IOMAP_UNWRITTEN) ^
|
|
|
|
(next->io_type == IOMAP_UNWRITTEN))
|
|
|
|
return false;
|
|
|
|
if (ioend->io_offset + ioend->io_size != next->io_offset)
|
|
|
|
return false;
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
/*
|
|
|
|
* Do not merge physically discontiguous ioends. The filesystem
|
|
|
|
* completion functions will have to iterate the physical
|
|
|
|
* discontiguities even if we merge the ioends at a logical level, so
|
|
|
|
* we don't gain anything by merging physical discontiguities here.
|
|
|
|
*
|
|
|
|
* We cannot use bio->bi_iter.bi_sector here as it is modified during
|
|
|
|
* submission so does not point to the start sector of the bio at
|
|
|
|
* completion.
|
|
|
|
*/
|
|
|
|
if (ioend->io_sector + (ioend->io_size >> 9) != next->io_sector)
|
|
|
|
return false;
|
2019-10-18 04:12:15 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2021-05-04 23:54:29 +08:00
|
|
|
iomap_ioend_try_merge(struct iomap_ioend *ioend, struct list_head *more_ioends)
|
2019-10-18 04:12:15 +08:00
|
|
|
{
|
|
|
|
struct iomap_ioend *next;
|
|
|
|
|
|
|
|
INIT_LIST_HEAD(&ioend->io_list);
|
|
|
|
|
|
|
|
while ((next = list_first_entry_or_null(more_ioends, struct iomap_ioend,
|
|
|
|
io_list))) {
|
|
|
|
if (!iomap_ioend_can_merge(ioend, next))
|
|
|
|
break;
|
|
|
|
list_move_tail(&next->io_list, &ioend->io_list);
|
|
|
|
ioend->io_size += next->io_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_ioend_try_merge);
|
|
|
|
|
|
|
|
static int
|
2021-04-09 02:28:34 +08:00
|
|
|
iomap_ioend_compare(void *priv, const struct list_head *a,
|
|
|
|
const struct list_head *b)
|
2019-10-18 04:12:15 +08:00
|
|
|
{
|
2019-10-18 04:12:20 +08:00
|
|
|
struct iomap_ioend *ia = container_of(a, struct iomap_ioend, io_list);
|
|
|
|
struct iomap_ioend *ib = container_of(b, struct iomap_ioend, io_list);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
if (ia->io_offset < ib->io_offset)
|
|
|
|
return -1;
|
2019-10-18 04:12:20 +08:00
|
|
|
if (ia->io_offset > ib->io_offset)
|
2019-10-18 04:12:15 +08:00
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
iomap_sort_ioends(struct list_head *ioend_list)
|
|
|
|
{
|
|
|
|
list_sort(NULL, ioend_list, iomap_ioend_compare);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_sort_ioends);
|
|
|
|
|
|
|
|
static void iomap_writepage_end_bio(struct bio *bio)
|
|
|
|
{
|
|
|
|
struct iomap_ioend *ioend = bio->bi_private;
|
|
|
|
|
|
|
|
iomap_finish_ioend(ioend, blk_status_to_errno(bio->bi_status));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Submit the final bio for an ioend.
|
|
|
|
*
|
|
|
|
* If @error is non-zero, it means that we have a situation where some part of
|
2021-08-03 05:46:31 +08:00
|
|
|
* the submission process has failed after we've marked pages for writeback
|
2019-10-18 04:12:15 +08:00
|
|
|
* and unlocked them. In this situation, we need to fail the bio instead of
|
|
|
|
* submitting it. This typically only happens on a filesystem shutdown.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
iomap_submit_ioend(struct iomap_writepage_ctx *wpc, struct iomap_ioend *ioend,
|
|
|
|
int error)
|
|
|
|
{
|
|
|
|
ioend->io_bio->bi_private = ioend;
|
|
|
|
ioend->io_bio->bi_end_io = iomap_writepage_end_bio;
|
|
|
|
|
|
|
|
if (wpc->ops->prepare_ioend)
|
|
|
|
error = wpc->ops->prepare_ioend(ioend, error);
|
|
|
|
if (error) {
|
|
|
|
/*
|
2021-08-03 05:46:31 +08:00
|
|
|
* If we're failing the IO now, just mark the ioend with an
|
2019-10-18 04:12:15 +08:00
|
|
|
* error and finish it. This will run IO completion immediately
|
|
|
|
* as there is only one reference to the ioend at this point in
|
|
|
|
* time.
|
|
|
|
*/
|
|
|
|
ioend->io_bio->bi_status = errno_to_blk_status(error);
|
|
|
|
bio_endio(ioend->io_bio);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
submit_bio(ioend->io_bio);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct iomap_ioend *
|
|
|
|
iomap_alloc_ioend(struct inode *inode, struct iomap_writepage_ctx *wpc,
|
|
|
|
loff_t offset, sector_t sector, struct writeback_control *wbc)
|
|
|
|
{
|
|
|
|
struct iomap_ioend *ioend;
|
|
|
|
struct bio *bio;
|
|
|
|
|
2022-01-24 17:11:03 +08:00
|
|
|
bio = bio_alloc_bioset(wpc->iomap.bdev, BIO_MAX_VECS,
|
|
|
|
REQ_OP_WRITE | wbc_to_write_flags(wbc),
|
|
|
|
GFP_NOFS, &iomap_ioend_bioset);
|
2019-10-18 04:12:15 +08:00
|
|
|
bio->bi_iter.bi_sector = sector;
|
|
|
|
wbc_init_bio(wbc, bio);
|
|
|
|
|
|
|
|
ioend = container_of(bio, struct iomap_ioend, io_inline_bio);
|
|
|
|
INIT_LIST_HEAD(&ioend->io_list);
|
|
|
|
ioend->io_type = wpc->iomap.type;
|
|
|
|
ioend->io_flags = wpc->iomap.flags;
|
|
|
|
ioend->io_inode = inode;
|
|
|
|
ioend->io_size = 0;
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
ioend->io_folios = 0;
|
2019-10-18 04:12:15 +08:00
|
|
|
ioend->io_offset = offset;
|
|
|
|
ioend->io_bio = bio;
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
ioend->io_sector = sector;
|
2019-10-18 04:12:15 +08:00
|
|
|
return ioend;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate a new bio, and chain the old bio to the new one.
|
|
|
|
*
|
2021-08-03 05:46:31 +08:00
|
|
|
* Note that we have to perform the chaining in this unintuitive order
|
2019-10-18 04:12:15 +08:00
|
|
|
* so that the bi_private linkage is set up in the right direction for the
|
|
|
|
* traversal in iomap_finish_ioend().
|
|
|
|
*/
|
|
|
|
static struct bio *
|
|
|
|
iomap_chain_bio(struct bio *prev)
|
|
|
|
{
|
|
|
|
struct bio *new;
|
|
|
|
|
2022-01-24 17:11:05 +08:00
|
|
|
new = bio_alloc(prev->bi_bdev, BIO_MAX_VECS, prev->bi_opf, GFP_NOFS);
|
|
|
|
bio_clone_blkg_association(new, prev);
|
2019-10-18 04:12:15 +08:00
|
|
|
new->bi_iter.bi_sector = bio_end_sector(prev);
|
|
|
|
|
|
|
|
bio_chain(prev, new);
|
|
|
|
bio_get(prev); /* for iomap_finish_ioend */
|
|
|
|
submit_bio(prev);
|
|
|
|
return new;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
iomap_can_add_to_ioend(struct iomap_writepage_ctx *wpc, loff_t offset,
|
|
|
|
sector_t sector)
|
|
|
|
{
|
|
|
|
if ((wpc->iomap.flags & IOMAP_F_SHARED) !=
|
|
|
|
(wpc->ioend->io_flags & IOMAP_F_SHARED))
|
|
|
|
return false;
|
|
|
|
if (wpc->iomap.type != wpc->ioend->io_type)
|
|
|
|
return false;
|
|
|
|
if (offset != wpc->ioend->io_offset + wpc->ioend->io_size)
|
|
|
|
return false;
|
|
|
|
if (sector != bio_end_sector(wpc->ioend->io_bio))
|
|
|
|
return false;
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
/*
|
|
|
|
* Limit ioend bio chain lengths to minimise IO completion latency. This
|
|
|
|
* also prevents long tight loops ending page writeback on all the
|
|
|
|
* folios in the ioend.
|
|
|
|
*/
|
|
|
|
if (wpc->ioend->io_folios >= IOEND_BATCH_SIZE)
|
|
|
|
return false;
|
2019-10-18 04:12:15 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Test to see if we have an existing ioend structure that we could append to
|
2021-08-03 05:46:31 +08:00
|
|
|
* first; otherwise finish off the current ioend and start another.
|
2019-10-18 04:12:15 +08:00
|
|
|
*/
|
|
|
|
static void
|
2021-11-03 00:45:12 +08:00
|
|
|
iomap_add_to_ioend(struct inode *inode, loff_t pos, struct folio *folio,
|
2019-10-18 04:12:15 +08:00
|
|
|
struct iomap_page *iop, struct iomap_writepage_ctx *wpc,
|
|
|
|
struct writeback_control *wbc, struct list_head *iolist)
|
|
|
|
{
|
2021-11-03 00:45:12 +08:00
|
|
|
sector_t sector = iomap_sector(&wpc->iomap, pos);
|
2019-10-18 04:12:15 +08:00
|
|
|
unsigned len = i_blocksize(inode);
|
2021-11-03 00:45:12 +08:00
|
|
|
size_t poff = offset_in_folio(folio, pos);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
2021-11-03 00:45:12 +08:00
|
|
|
if (!wpc->ioend || !iomap_can_add_to_ioend(wpc, pos, sector)) {
|
2019-10-18 04:12:15 +08:00
|
|
|
if (wpc->ioend)
|
|
|
|
list_add(&wpc->ioend->io_list, iolist);
|
2021-11-03 00:45:12 +08:00
|
|
|
wpc->ioend = iomap_alloc_ioend(inode, wpc, pos, sector, wbc);
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
|
2021-11-03 00:45:12 +08:00
|
|
|
if (!bio_add_folio(wpc->ioend->io_bio, folio, len, poff)) {
|
2021-08-03 05:43:43 +08:00
|
|
|
wpc->ioend->io_bio = iomap_chain_bio(wpc->ioend->io_bio);
|
2021-11-03 00:45:12 +08:00
|
|
|
bio_add_folio(wpc->ioend->io_bio, folio, len, poff);
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
|
2021-08-03 05:43:43 +08:00
|
|
|
if (iop)
|
|
|
|
atomic_add(len, &iop->write_bytes_pending);
|
2019-10-18 04:12:15 +08:00
|
|
|
wpc->ioend->io_size += len;
|
2021-11-03 00:45:12 +08:00
|
|
|
wbc_account_cgroup_owner(wbc, &folio->page, len);
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We implement an immediate ioend submission policy here to avoid needing to
|
|
|
|
* chain multiple ioends and hence nest mempool allocations which can violate
|
2021-08-03 05:46:31 +08:00
|
|
|
* the forward progress guarantees we need to provide. The current ioend we're
|
|
|
|
* adding blocks to is cached in the writepage context, and if the new block
|
|
|
|
* doesn't append to the cached ioend, it will create a new ioend and cache that
|
2019-10-18 04:12:15 +08:00
|
|
|
* instead.
|
|
|
|
*
|
|
|
|
* If a new ioend is created and cached, the old ioend is returned and queued
|
|
|
|
* locally for submission once the entire page is processed or an error has been
|
|
|
|
* detected. While ioends are submitted immediately after they are completed,
|
|
|
|
* batching optimisations are provided by higher level block plugging.
|
|
|
|
*
|
|
|
|
* At the end of a writeback pass, there will be a cached ioend remaining on the
|
|
|
|
* writepage context that the caller will need to submit.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
iomap_writepage_map(struct iomap_writepage_ctx *wpc,
|
|
|
|
struct writeback_control *wbc, struct inode *inode,
|
2021-11-03 00:45:12 +08:00
|
|
|
struct folio *folio, u64 end_pos)
|
2019-10-18 04:12:15 +08:00
|
|
|
{
|
2022-06-24 01:51:47 +08:00
|
|
|
struct iomap_page *iop = iomap_page_create(inode, folio, 0);
|
2019-10-18 04:12:15 +08:00
|
|
|
struct iomap_ioend *ioend, *next;
|
|
|
|
unsigned len = i_blocksize(inode);
|
2021-11-02 22:51:55 +08:00
|
|
|
unsigned nblocks = i_blocks_per_folio(inode, folio);
|
|
|
|
u64 pos = folio_pos(folio);
|
2019-10-18 04:12:15 +08:00
|
|
|
int error = 0, count = 0, i;
|
|
|
|
LIST_HEAD(submit_list);
|
|
|
|
|
2020-09-21 23:58:41 +08:00
|
|
|
WARN_ON_ONCE(iop && atomic_read(&iop->write_bytes_pending) != 0);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
/*
|
2021-11-02 22:51:55 +08:00
|
|
|
* Walk through the folio to find areas to write back. If we
|
|
|
|
* run off the end of the current map or find the current map
|
|
|
|
* invalid, grab a new one.
|
2019-10-18 04:12:15 +08:00
|
|
|
*/
|
2021-11-02 22:51:55 +08:00
|
|
|
for (i = 0; i < nblocks && pos < end_pos; i++, pos += len) {
|
2019-10-18 04:12:15 +08:00
|
|
|
if (iop && !test_bit(i, iop->uptodate))
|
|
|
|
continue;
|
|
|
|
|
2021-11-02 22:51:55 +08:00
|
|
|
error = wpc->ops->map_blocks(wpc, inode, pos);
|
2019-10-18 04:12:15 +08:00
|
|
|
if (error)
|
|
|
|
break;
|
2022-10-01 08:02:32 +08:00
|
|
|
trace_iomap_writepage_map(inode, &wpc->iomap);
|
2019-10-18 04:12:17 +08:00
|
|
|
if (WARN_ON_ONCE(wpc->iomap.type == IOMAP_INLINE))
|
|
|
|
continue;
|
2019-10-18 04:12:15 +08:00
|
|
|
if (wpc->iomap.type == IOMAP_HOLE)
|
|
|
|
continue;
|
2021-11-03 00:45:12 +08:00
|
|
|
iomap_add_to_ioend(inode, pos, folio, iop, wpc, wbc,
|
2019-10-18 04:12:15 +08:00
|
|
|
&submit_list);
|
|
|
|
count++;
|
|
|
|
}
|
xfs, iomap: limit individual ioend chain lengths in writeback
Trond Myklebust reported soft lockups in XFS IO completion such as
this:
watchdog: BUG: soft lockup - CPU#12 stuck for 23s! [kworker/12:1:3106]
CPU: 12 PID: 3106 Comm: kworker/12:1 Not tainted 4.18.0-305.10.2.el8_4.x86_64 #1
Workqueue: xfs-conv/md127 xfs_end_io [xfs]
RIP: 0010:_raw_spin_unlock_irqrestore+0x11/0x20
Call Trace:
wake_up_page_bit+0x8a/0x110
iomap_finish_ioend+0xd7/0x1c0
iomap_finish_ioends+0x7f/0xb0
xfs_end_ioend+0x6b/0x100 [xfs]
xfs_end_io+0xb9/0xe0 [xfs]
process_one_work+0x1a7/0x360
worker_thread+0x1fa/0x390
kthread+0x116/0x130
ret_from_fork+0x35/0x40
Ioends are processed as an atomic completion unit when all the
chained bios in the ioend have completed their IO. Logically
contiguous ioends can also be merged and completed as a single,
larger unit. Both of these things can be problematic as both the
bio chains per ioend and the size of the merged ioends processed as
a single completion are both unbound.
If we have a large sequential dirty region in the page cache,
write_cache_pages() will keep feeding us sequential pages and we
will keep mapping them into ioends and bios until we get a dirty
page at a non-sequential file offset. These large sequential runs
can will result in bio and ioend chaining to optimise the io
patterns. The pages iunder writeback are pinned within these chains
until the submission chaining is broken, allowing the entire chain
to be completed. This can result in huge chains being processed
in IO completion context.
We get deep bio chaining if we have large contiguous physical
extents. We will keep adding pages to the current bio until it is
full, then we'll chain a new bio to keep adding pages for writeback.
Hence we can build bio chains that map millions of pages and tens of
gigabytes of RAM if the page cache contains big enough contiguous
dirty file regions. This long bio chain pins those pages until the
final bio in the chain completes and the ioend can iterate all the
chained bios and complete them.
OTOH, if we have a physically fragmented file, we end up submitting
one ioend per physical fragment that each have a small bio or bio
chain attached to them. We do not chain these at IO submission time,
but instead we chain them at completion time based on file
offset via iomap_ioend_try_merge(). Hence we can end up with unbound
ioend chains being built via completion merging.
XFS can then do COW remapping or unwritten extent conversion on that
merged chain, which involves walking an extent fragment at a time
and running a transaction to modify the physical extent information.
IOWs, we merge all the discontiguous ioends together into a
contiguous file range, only to then process them individually as
discontiguous extents.
This extent manipulation is computationally expensive and can run in
a tight loop, so merging logically contiguous but physically
discontigous ioends gains us nothing except for hiding the fact the
fact we broke the ioends up into individual physical extents at
submission and then need to loop over those individual physical
extents at completion.
Hence we need to have mechanisms to limit ioend sizes and
to break up completion processing of large merged ioend chains:
1. bio chains per ioend need to be bound in length. Pure overwrites
go straight to iomap_finish_ioend() in softirq context with the
exact bio chain attached to the ioend by submission. Hence the only
way to prevent long holdoffs here is to bound ioend submission
sizes because we can't reschedule in softirq context.
2. iomap_finish_ioends() has to handle unbound merged ioend chains
correctly. This relies on any one call to iomap_finish_ioend() being
bound in runtime so that cond_resched() can be issued regularly as
the long ioend chain is processed. i.e. this relies on mechanism #1
to limit individual ioend sizes to work correctly.
3. filesystems have to loop over the merged ioends to process
physical extent manipulations. This means they can loop internally,
and so we break merging at physical extent boundaries so the
filesystem can easily insert reschedule points between individual
extent manipulations.
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reported-and-tested-by: Trond Myklebust <trondmy@hammerspace.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-01-27 01:19:20 +08:00
|
|
|
if (count)
|
|
|
|
wpc->ioend->io_folios++;
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
WARN_ON_ONCE(!wpc->ioend && !list_empty(&submit_list));
|
2021-11-03 00:45:12 +08:00
|
|
|
WARN_ON_ONCE(!folio_test_locked(folio));
|
|
|
|
WARN_ON_ONCE(folio_test_writeback(folio));
|
|
|
|
WARN_ON_ONCE(folio_test_dirty(folio));
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We cannot cancel the ioend directly here on error. We may have
|
|
|
|
* already set other pages under writeback and hence we have to run I/O
|
|
|
|
* completion to mark the error state of the pages under writeback
|
|
|
|
* appropriately.
|
|
|
|
*/
|
|
|
|
if (unlikely(error)) {
|
2020-10-30 05:30:48 +08:00
|
|
|
/*
|
|
|
|
* Let the filesystem know what portion of the current page
|
2021-08-03 05:46:31 +08:00
|
|
|
* failed to map. If the page hasn't been added to ioend, it
|
2020-10-30 05:30:48 +08:00
|
|
|
* won't be affected by I/O completion and we must unlock it
|
|
|
|
* now.
|
|
|
|
*/
|
2021-07-30 21:56:05 +08:00
|
|
|
if (wpc->ops->discard_folio)
|
2021-11-02 22:51:55 +08:00
|
|
|
wpc->ops->discard_folio(folio, pos);
|
2019-10-18 04:12:15 +08:00
|
|
|
if (!count) {
|
2021-11-03 00:45:12 +08:00
|
|
|
folio_unlock(folio);
|
2019-10-18 04:12:15 +08:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-03 00:45:12 +08:00
|
|
|
folio_start_writeback(folio);
|
|
|
|
folio_unlock(folio);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
/*
|
2021-08-03 05:46:31 +08:00
|
|
|
* Preserve the original error if there was one; catch
|
2019-10-18 04:12:15 +08:00
|
|
|
* submission errors here and propagate into subsequent ioend
|
|
|
|
* submissions.
|
|
|
|
*/
|
|
|
|
list_for_each_entry_safe(ioend, next, &submit_list, io_list) {
|
|
|
|
int error2;
|
|
|
|
|
|
|
|
list_del_init(&ioend->io_list);
|
|
|
|
error2 = iomap_submit_ioend(wpc, ioend, error);
|
|
|
|
if (error2 && !error)
|
|
|
|
error = error2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We can end up here with no error and nothing to write only if we race
|
|
|
|
* with a partial page truncate on a sub-page block sized filesystem.
|
|
|
|
*/
|
|
|
|
if (!count)
|
2021-11-03 00:45:12 +08:00
|
|
|
folio_end_writeback(folio);
|
2019-10-18 04:12:15 +08:00
|
|
|
done:
|
iomap: iomap: fix memory corruption when recording errors during writeback
Every now and then I see this crash on arm64:
Unable to handle kernel NULL pointer dereference at virtual address 00000000000000f8
Buffer I/O error on dev dm-0, logical block 8733687, async page read
Mem abort info:
ESR = 0x0000000096000006
EC = 0x25: DABT (current EL), IL = 32 bits
SET = 0, FnV = 0
EA = 0, S1PTW = 0
FSC = 0x06: level 2 translation fault
Data abort info:
ISV = 0, ISS = 0x00000006
CM = 0, WnR = 0
user pgtable: 64k pages, 42-bit VAs, pgdp=0000000139750000
[00000000000000f8] pgd=0000000000000000, p4d=0000000000000000, pud=0000000000000000, pmd=0000000000000000
Internal error: Oops: 96000006 [#1] PREEMPT SMP
Buffer I/O error on dev dm-0, logical block 8733688, async page read
Dumping ftrace buffer:
Buffer I/O error on dev dm-0, logical block 8733689, async page read
(ftrace buffer empty)
XFS (dm-0): log I/O error -5
Modules linked in: dm_thin_pool dm_persistent_data
XFS (dm-0): Metadata I/O Error (0x1) detected at xfs_trans_read_buf_map+0x1ec/0x590 [xfs] (fs/xfs/xfs_trans_buf.c:296).
dm_bio_prison
XFS (dm-0): Please unmount the filesystem and rectify the problem(s)
XFS (dm-0): xfs_imap_lookup: xfs_ialloc_read_agi() returned error -5, agno 0
dm_bufio dm_log_writes xfs nft_chain_nat xt_REDIRECT nf_nat nf_conntrack nf_defrag_ipv6 nf_defrag_ipv4 ip6t_REJECT
potentially unexpected fatal signal 6.
nf_reject_ipv6
potentially unexpected fatal signal 6.
ipt_REJECT nf_reject_ipv4
CPU: 1 PID: 122166 Comm: fsstress Tainted: G W 6.0.0-rc5-djwa #rc5 3004c9f1de887ebae86015f2677638ce51ee7
rpcsec_gss_krb5 auth_rpcgss xt_tcpudp ip_set_hash_ip ip_set_hash_net xt_set nft_compat ip_set_hash_mac ip_set nf_tables
Hardware name: QEMU KVM Virtual Machine, BIOS 1.5.1 06/16/2021
pstate: 60001000 (nZCv daif -PAN -UAO -TCO -DIT +SSBS BTYPE=--)
ip_tables
pc : 000003fd6d7df200
x_tables
lr : 000003fd6d7df1ec
overlay nfsv4
CPU: 0 PID: 54031 Comm: u4:3 Tainted: G W 6.0.0-rc5-djwa #rc5 3004c9f1de887ebae86015f2677638ce51ee7405
Hardware name: QEMU KVM Virtual Machine, BIOS 1.5.1 06/16/2021
Workqueue: writeback wb_workfn
sp : 000003ffd9522fd0
(flush-253:0)
pstate: 60401005 (nZCv daif +PAN -UAO -TCO -DIT +SSBS BTYPE=--)
pc : errseq_set+0x1c/0x100
x29: 000003ffd9522fd0 x28: 0000000000000023 x27: 000002acefeb6780
x26: 0000000000000005 x25: 0000000000000001 x24: 0000000000000000
x23: 00000000ffffffff x22: 0000000000000005
lr : __filemap_set_wb_err+0x24/0xe0
x21: 0000000000000006
sp : fffffe000f80f760
x29: fffffe000f80f760 x28: 0000000000000003 x27: fffffe000f80f9f8
x26: 0000000002523000 x25: 00000000fffffffb x24: fffffe000f80f868
x23: fffffe000f80fbb0 x22: fffffc0180c26a78 x21: 0000000002530000
x20: 0000000000000000 x19: 0000000000000000 x18: 0000000000000000
x17: 0000000000000000 x16: 0000000000000000 x15: 0000000000000000
x14: 0000000000000001 x13: 0000000000470af3 x12: fffffc0058f70000
x11: 0000000000000040 x10: 0000000000001b20 x9 : fffffe000836b288
x8 : fffffc00eb9fd480 x7 : 0000000000f83659 x6 : 0000000000000000
x5 : 0000000000000869 x4 : 0000000000000005 x3 : 00000000000000f8
x20: 000003fd6d740020 x19: 000000000001dd36 x18: 0000000000000001
x17: 000003fd6d78704c x16: 0000000000000001 x15: 000002acfac87668
x2 : 0000000000000ffa x1 : 00000000fffffffb x0 : 00000000000000f8
Call trace:
errseq_set+0x1c/0x100
__filemap_set_wb_err+0x24/0xe0
iomap_do_writepage+0x5e4/0xd5c
write_cache_pages+0x208/0x674
iomap_writepages+0x34/0x60
xfs_vm_writepages+0x8c/0xcc [xfs 7a861f39c43631f15d3a5884246ba5035d4ca78b]
x14: 0000000000000000 x13: 2064656e72757465 x12: 0000000000002180
x11: 000003fd6d8a82d0 x10: 0000000000000000 x9 : 000003fd6d8ae288
x8 : 0000000000000083 x7 : 00000000ffffffff x6 : 00000000ffffffee
x5 : 00000000fbad2887 x4 : 000003fd6d9abb58 x3 : 000003fd6d740020
x2 : 0000000000000006 x1 : 000000000001dd36 x0 : 0000000000000000
CPU: 1 PID: 122167 Comm: fsstress Tainted: G W 6.0.0-rc5-djwa #rc5 3004c9f1de887ebae86015f2677638ce51ee7
do_writepages+0x90/0x1c4
__writeback_single_inode+0x4c/0x4ac
Hardware name: QEMU KVM Virtual Machine, BIOS 1.5.1 06/16/2021
writeback_sb_inodes+0x214/0x4ac
wb_writeback+0xf4/0x3b0
pstate: 60001000 (nZCv daif -PAN -UAO -TCO -DIT +SSBS BTYPE=--)
wb_workfn+0xfc/0x580
process_one_work+0x1e8/0x480
pc : 000003fd6d7df200
worker_thread+0x78/0x430
This crash is a result of iomap_writepage_map encountering some sort of
error during writeback and wanting to set that error code in the file
mapping so that fsync will report it. Unfortunately, the code
dereferences folio->mapping after unlocking the folio, which means that
another thread could have removed the page from the page cache
(writeback doesn't hold the invalidation lock) and give it to somebody
else.
At best we crash the system like above; at worst, we corrupt memory or
set an error on some other unsuspecting file while failing to record the
problems with *this* file. Regardless, fix the problem by reporting the
error to the inode mapping.
NOTE: Commit 598ecfbaa742 lifted the XFS writeback code to iomap, so
this fix should be backported to XFS in the 4.6-5.4 kernels in addition
to iomap in the 5.5-5.19 kernels.
Fixes: e735c0079465 ("iomap: Convert iomap_add_to_ioend() to take a folio") # 5.17 onward
Fixes: 598ecfbaa742 ("iomap: lift the xfs writeback code to iomap") # 5.5-5.16, needs backporting
Fixes: 150d5be09ce4 ("xfs: remove xfs_cancel_ioend") # 4.6-5.4, needs backporting
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
Reviewed-by: Matthew Wilcox (Oracle) <willy@infradead.org>
2022-10-01 08:02:32 +08:00
|
|
|
mapping_set_error(inode->i_mapping, error);
|
2019-10-18 04:12:15 +08:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write out a dirty page.
|
|
|
|
*
|
2021-08-03 05:46:31 +08:00
|
|
|
* For delalloc space on the page, we need to allocate space and flush it.
|
|
|
|
* For unwritten space on the page, we need to start the conversion to
|
2019-10-18 04:12:15 +08:00
|
|
|
* regular allocated space.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
iomap_do_writepage(struct page *page, struct writeback_control *wbc, void *data)
|
|
|
|
{
|
2021-11-03 00:45:12 +08:00
|
|
|
struct folio *folio = page_folio(page);
|
2019-10-18 04:12:15 +08:00
|
|
|
struct iomap_writepage_ctx *wpc = data;
|
2021-11-03 00:45:12 +08:00
|
|
|
struct inode *inode = folio->mapping->host;
|
2021-11-02 23:41:16 +08:00
|
|
|
u64 end_pos, isize;
|
2019-10-18 04:12:15 +08:00
|
|
|
|
2021-11-03 00:45:12 +08:00
|
|
|
trace_iomap_writepage(inode, folio_pos(folio), folio_size(folio));
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
/*
|
2021-11-03 00:45:12 +08:00
|
|
|
* Refuse to write the folio out if we're called from reclaim context.
|
2019-10-18 04:12:15 +08:00
|
|
|
*
|
|
|
|
* This avoids stack overflows when called from deeply used stacks in
|
|
|
|
* random callers for direct reclaim or memcg reclaim. We explicitly
|
|
|
|
* allow reclaim from kswapd as the stack usage there is relatively low.
|
|
|
|
*
|
|
|
|
* This should never happen except in the case of a VM regression so
|
|
|
|
* warn about it.
|
|
|
|
*/
|
|
|
|
if (WARN_ON_ONCE((current->flags & (PF_MEMALLOC|PF_KSWAPD)) ==
|
|
|
|
PF_MEMALLOC))
|
|
|
|
goto redirty;
|
|
|
|
|
|
|
|
/*
|
2021-11-03 00:45:12 +08:00
|
|
|
* Is this folio beyond the end of the file?
|
2019-10-18 04:12:15 +08:00
|
|
|
*
|
2021-11-03 00:45:12 +08:00
|
|
|
* The folio index is less than the end_index, adjust the end_pos
|
|
|
|
* to the highest offset that this folio should represent.
|
2019-10-18 04:12:15 +08:00
|
|
|
* -----------------------------------------------------
|
|
|
|
* | file mapping | <EOF> |
|
|
|
|
* -----------------------------------------------------
|
|
|
|
* | Page ... | Page N-2 | Page N-1 | Page N | |
|
|
|
|
* ^--------------------------------^----------|--------
|
|
|
|
* | desired writeback range | see else |
|
|
|
|
* ---------------------------------^------------------|
|
|
|
|
*/
|
2021-11-02 23:41:16 +08:00
|
|
|
isize = i_size_read(inode);
|
2021-11-03 00:45:12 +08:00
|
|
|
end_pos = folio_pos(folio) + folio_size(folio);
|
2021-11-02 23:41:16 +08:00
|
|
|
if (end_pos > isize) {
|
2019-10-18 04:12:15 +08:00
|
|
|
/*
|
|
|
|
* Check whether the page to write out is beyond or straddles
|
|
|
|
* i_size or not.
|
|
|
|
* -------------------------------------------------------
|
|
|
|
* | file mapping | <EOF> |
|
|
|
|
* -------------------------------------------------------
|
|
|
|
* | Page ... | Page N-2 | Page N-1 | Page N | Beyond |
|
|
|
|
* ^--------------------------------^-----------|---------
|
|
|
|
* | | Straddles |
|
|
|
|
* ---------------------------------^-----------|--------|
|
|
|
|
*/
|
2021-11-03 00:45:12 +08:00
|
|
|
size_t poff = offset_in_folio(folio, isize);
|
2021-11-02 23:41:16 +08:00
|
|
|
pgoff_t end_index = isize >> PAGE_SHIFT;
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
/*
|
iomap: skip pages past eof in iomap_do_writepage()
iomap_do_writepage() sends pages past i_size through
folio_redirty_for_writepage(), which normally isn't a problem because
truncate and friends clean them very quickly.
When the system has cgroups configured, we can end up in situations
where one cgroup has almost no dirty pages at all, and other cgroups
consume the entire background dirty limit. This is especially common in
our XFS workloads in production because they have cgroups using O_DIRECT
for almost all of the IO mixed in with cgroups that do more traditional
buffered IO work.
We've hit storms where the redirty path hits millions of times in a few
seconds, on all a single file that's only ~40 pages long. This leads to
long tail latencies for file writes because the pdflush workers are
hogging the CPU from some kworkers bound to the same CPU.
Reproducing this on 5.18 was tricky because 869ae85dae ("xfs: flush new
eof page on truncate...") ends up writing/waiting most of these dirty pages
before truncate gets a chance to wait on them.
The actual repro looks like this:
/*
* run me in a cgroup all alone. Start a second cgroup with dd
* streaming IO into the block device.
*/
int main(int ac, char **av) {
int fd;
int ret;
char buf[BUFFER_SIZE];
char *filename = av[1];
memset(buf, 0, BUFFER_SIZE);
if (ac != 2) {
fprintf(stderr, "usage: looper filename\n");
exit(1);
}
fd = open(filename, O_WRONLY | O_CREAT, 0600);
if (fd < 0) {
err(errno, "failed to open");
}
fprintf(stderr, "looping on %s\n", filename);
while(1) {
/*
* skip past page 0 so truncate doesn't write and wait
* on our extent before changing i_size
*/
ret = lseek(fd, 8192, SEEK_SET);
if (ret < 0)
err(errno, "lseek");
ret = write(fd, buf, BUFFER_SIZE);
if (ret != BUFFER_SIZE)
err(errno, "write failed");
/* start IO so truncate has to wait after i_size is 0 */
ret = sync_file_range(fd, 16384, 4095, SYNC_FILE_RANGE_WRITE);
if (ret < 0)
err(errno, "sync_file_range");
ret = ftruncate(fd, 0);
if (ret < 0)
err(errno, "truncate");
usleep(1000);
}
}
And this bpftrace script will show when you've hit a redirty storm:
kretprobe:xfs_vm_writepages {
delete(@dirty[pid]);
}
kprobe:xfs_vm_writepages {
@dirty[pid] = 1;
}
kprobe:folio_redirty_for_writepage /@dirty[pid] > 0/ {
$inode = ((struct folio *)arg1)->mapping->host->i_ino;
@inodes[$inode] = count();
@redirty++;
if (@redirty > 90000) {
printf("inode %d redirty was %d", $inode, @redirty);
exit();
}
}
This patch has the same number of failures on xfstests as unpatched 5.18:
Failures: generic/648 xfs/019 xfs/050 xfs/168 xfs/299 xfs/348 xfs/506
xfs/543
I also ran it through a long stress of multiple fsx processes hammering.
(Johannes Weiner did significant tracing and debugging on this as well)
Signed-off-by: Chris Mason <clm@fb.com>
Co-authored-by: Johannes Weiner <hannes@cmpxchg.org>
Reviewed-by: Johannes Weiner <hannes@cmpxchg.org>
Reported-by: Domas Mituzas <domas@fb.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-07-01 00:52:57 +08:00
|
|
|
* Skip the page if it's fully outside i_size, e.g.
|
|
|
|
* due to a truncate operation that's in progress. We've
|
|
|
|
* cleaned this page and truncate will finish things off for
|
|
|
|
* us.
|
2019-10-18 04:12:15 +08:00
|
|
|
*
|
2021-08-03 05:46:31 +08:00
|
|
|
* Note that the end_index is unsigned long. If the given
|
|
|
|
* offset is greater than 16TB on a 32-bit system then if we
|
|
|
|
* checked if the page is fully outside i_size with
|
|
|
|
* "if (page->index >= end_index + 1)", "end_index + 1" would
|
|
|
|
* overflow and evaluate to 0. Hence this page would be
|
|
|
|
* redirtied and written out repeatedly, which would result in
|
|
|
|
* an infinite loop; the user program performing this operation
|
|
|
|
* would hang. Instead, we can detect this situation by
|
|
|
|
* checking if the page is totally beyond i_size or if its
|
2019-10-18 04:12:15 +08:00
|
|
|
* offset is just equal to the EOF.
|
|
|
|
*/
|
2021-11-03 00:45:12 +08:00
|
|
|
if (folio->index > end_index ||
|
|
|
|
(folio->index == end_index && poff == 0))
|
iomap: skip pages past eof in iomap_do_writepage()
iomap_do_writepage() sends pages past i_size through
folio_redirty_for_writepage(), which normally isn't a problem because
truncate and friends clean them very quickly.
When the system has cgroups configured, we can end up in situations
where one cgroup has almost no dirty pages at all, and other cgroups
consume the entire background dirty limit. This is especially common in
our XFS workloads in production because they have cgroups using O_DIRECT
for almost all of the IO mixed in with cgroups that do more traditional
buffered IO work.
We've hit storms where the redirty path hits millions of times in a few
seconds, on all a single file that's only ~40 pages long. This leads to
long tail latencies for file writes because the pdflush workers are
hogging the CPU from some kworkers bound to the same CPU.
Reproducing this on 5.18 was tricky because 869ae85dae ("xfs: flush new
eof page on truncate...") ends up writing/waiting most of these dirty pages
before truncate gets a chance to wait on them.
The actual repro looks like this:
/*
* run me in a cgroup all alone. Start a second cgroup with dd
* streaming IO into the block device.
*/
int main(int ac, char **av) {
int fd;
int ret;
char buf[BUFFER_SIZE];
char *filename = av[1];
memset(buf, 0, BUFFER_SIZE);
if (ac != 2) {
fprintf(stderr, "usage: looper filename\n");
exit(1);
}
fd = open(filename, O_WRONLY | O_CREAT, 0600);
if (fd < 0) {
err(errno, "failed to open");
}
fprintf(stderr, "looping on %s\n", filename);
while(1) {
/*
* skip past page 0 so truncate doesn't write and wait
* on our extent before changing i_size
*/
ret = lseek(fd, 8192, SEEK_SET);
if (ret < 0)
err(errno, "lseek");
ret = write(fd, buf, BUFFER_SIZE);
if (ret != BUFFER_SIZE)
err(errno, "write failed");
/* start IO so truncate has to wait after i_size is 0 */
ret = sync_file_range(fd, 16384, 4095, SYNC_FILE_RANGE_WRITE);
if (ret < 0)
err(errno, "sync_file_range");
ret = ftruncate(fd, 0);
if (ret < 0)
err(errno, "truncate");
usleep(1000);
}
}
And this bpftrace script will show when you've hit a redirty storm:
kretprobe:xfs_vm_writepages {
delete(@dirty[pid]);
}
kprobe:xfs_vm_writepages {
@dirty[pid] = 1;
}
kprobe:folio_redirty_for_writepage /@dirty[pid] > 0/ {
$inode = ((struct folio *)arg1)->mapping->host->i_ino;
@inodes[$inode] = count();
@redirty++;
if (@redirty > 90000) {
printf("inode %d redirty was %d", $inode, @redirty);
exit();
}
}
This patch has the same number of failures on xfstests as unpatched 5.18:
Failures: generic/648 xfs/019 xfs/050 xfs/168 xfs/299 xfs/348 xfs/506
xfs/543
I also ran it through a long stress of multiple fsx processes hammering.
(Johannes Weiner did significant tracing and debugging on this as well)
Signed-off-by: Chris Mason <clm@fb.com>
Co-authored-by: Johannes Weiner <hannes@cmpxchg.org>
Reviewed-by: Johannes Weiner <hannes@cmpxchg.org>
Reported-by: Domas Mituzas <domas@fb.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-07-01 00:52:57 +08:00
|
|
|
goto unlock;
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The page straddles i_size. It must be zeroed out on each
|
|
|
|
* and every writepage invocation because it may be mmapped.
|
|
|
|
* "A file is mapped in multiples of the page size. For a file
|
|
|
|
* that is not a multiple of the page size, the remaining
|
|
|
|
* memory is zeroed when mapped, and writes to that region are
|
|
|
|
* not written out to the file."
|
|
|
|
*/
|
2021-11-03 00:45:12 +08:00
|
|
|
folio_zero_segment(folio, poff, folio_size(folio));
|
2021-11-02 23:41:16 +08:00
|
|
|
end_pos = isize;
|
2019-10-18 04:12:15 +08:00
|
|
|
}
|
|
|
|
|
2021-11-03 00:45:12 +08:00
|
|
|
return iomap_writepage_map(wpc, wbc, inode, folio, end_pos);
|
2019-10-18 04:12:15 +08:00
|
|
|
|
|
|
|
redirty:
|
2021-11-03 00:45:12 +08:00
|
|
|
folio_redirty_for_writepage(wbc, folio);
|
iomap: skip pages past eof in iomap_do_writepage()
iomap_do_writepage() sends pages past i_size through
folio_redirty_for_writepage(), which normally isn't a problem because
truncate and friends clean them very quickly.
When the system has cgroups configured, we can end up in situations
where one cgroup has almost no dirty pages at all, and other cgroups
consume the entire background dirty limit. This is especially common in
our XFS workloads in production because they have cgroups using O_DIRECT
for almost all of the IO mixed in with cgroups that do more traditional
buffered IO work.
We've hit storms where the redirty path hits millions of times in a few
seconds, on all a single file that's only ~40 pages long. This leads to
long tail latencies for file writes because the pdflush workers are
hogging the CPU from some kworkers bound to the same CPU.
Reproducing this on 5.18 was tricky because 869ae85dae ("xfs: flush new
eof page on truncate...") ends up writing/waiting most of these dirty pages
before truncate gets a chance to wait on them.
The actual repro looks like this:
/*
* run me in a cgroup all alone. Start a second cgroup with dd
* streaming IO into the block device.
*/
int main(int ac, char **av) {
int fd;
int ret;
char buf[BUFFER_SIZE];
char *filename = av[1];
memset(buf, 0, BUFFER_SIZE);
if (ac != 2) {
fprintf(stderr, "usage: looper filename\n");
exit(1);
}
fd = open(filename, O_WRONLY | O_CREAT, 0600);
if (fd < 0) {
err(errno, "failed to open");
}
fprintf(stderr, "looping on %s\n", filename);
while(1) {
/*
* skip past page 0 so truncate doesn't write and wait
* on our extent before changing i_size
*/
ret = lseek(fd, 8192, SEEK_SET);
if (ret < 0)
err(errno, "lseek");
ret = write(fd, buf, BUFFER_SIZE);
if (ret != BUFFER_SIZE)
err(errno, "write failed");
/* start IO so truncate has to wait after i_size is 0 */
ret = sync_file_range(fd, 16384, 4095, SYNC_FILE_RANGE_WRITE);
if (ret < 0)
err(errno, "sync_file_range");
ret = ftruncate(fd, 0);
if (ret < 0)
err(errno, "truncate");
usleep(1000);
}
}
And this bpftrace script will show when you've hit a redirty storm:
kretprobe:xfs_vm_writepages {
delete(@dirty[pid]);
}
kprobe:xfs_vm_writepages {
@dirty[pid] = 1;
}
kprobe:folio_redirty_for_writepage /@dirty[pid] > 0/ {
$inode = ((struct folio *)arg1)->mapping->host->i_ino;
@inodes[$inode] = count();
@redirty++;
if (@redirty > 90000) {
printf("inode %d redirty was %d", $inode, @redirty);
exit();
}
}
This patch has the same number of failures on xfstests as unpatched 5.18:
Failures: generic/648 xfs/019 xfs/050 xfs/168 xfs/299 xfs/348 xfs/506
xfs/543
I also ran it through a long stress of multiple fsx processes hammering.
(Johannes Weiner did significant tracing and debugging on this as well)
Signed-off-by: Chris Mason <clm@fb.com>
Co-authored-by: Johannes Weiner <hannes@cmpxchg.org>
Reviewed-by: Johannes Weiner <hannes@cmpxchg.org>
Reported-by: Domas Mituzas <domas@fb.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
2022-07-01 00:52:57 +08:00
|
|
|
unlock:
|
2021-11-03 00:45:12 +08:00
|
|
|
folio_unlock(folio);
|
2019-10-18 04:12:15 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
iomap_writepages(struct address_space *mapping, struct writeback_control *wbc,
|
|
|
|
struct iomap_writepage_ctx *wpc,
|
|
|
|
const struct iomap_writeback_ops *ops)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
wpc->ops = ops;
|
|
|
|
ret = write_cache_pages(mapping, wbc, iomap_do_writepage, wpc);
|
|
|
|
if (!wpc->ioend)
|
|
|
|
return ret;
|
|
|
|
return iomap_submit_ioend(wpc, wpc->ioend, ret);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(iomap_writepages);
|
|
|
|
|
|
|
|
static int __init iomap_init(void)
|
|
|
|
{
|
|
|
|
return bioset_init(&iomap_ioend_bioset, 4 * (PAGE_SIZE / SECTOR_SIZE),
|
|
|
|
offsetof(struct iomap_ioend, io_inline_bio),
|
|
|
|
BIOSET_NEED_BVECS);
|
|
|
|
}
|
|
|
|
fs_initcall(iomap_init);
|