On 21 Jun 2018, at 23:09, Dave Chinner wrote:
From: Dave Chinner <dchinner@xxxxxxxxxx>
We've recently seen a workload on XFS filesystems with a repeatable
deadlock between background writeback and a multi-process
application doing concurrent writes and fsyncs to a small range of a
file.
The underlying cause of the problem here is that range_cyclic
writeback is processing pages in descending index order as we
hold higher index pages in a structure controlled from above
write_cache_pages(). The write_cache_pages() caller needs to
be able to submit these pages for IO before write_cache_pages
restarts writeback at mapping index 0 to avoid wcp inverting the
page lock/writeback wait order.
generic_writepages() is not susceptible to this bug as it has no
private context held across write_cache_pages() - filesystems using
this infrastructure always submit pages in ->writepage immediately
and so there is no problem with range_cyclic going back to mapping
index 0.
However:
mpage_writepages() has a private bio context,
exofs_writepages() has page_collect
fuse_writepages() has fuse_fill_wb_data
nfs_writepages() has nfs_pageio_descriptor
xfs_vm_writepages() has xfs_writepage_ctx
All of these ->writepages implementations can hold pages under
writeback in their private structures until write_cache_pages()
returns, and hence they are all susceptible to this deadlock.
Also worth noting is that ext4 has it's own bastardised version of
write_cache_pages() and so it /may/ have an equivalent deadlock. I
looked at the code long enough to understand that it has a similar
retry loop for range_cyclic writeback reaching the end of the file
and then promptly ran away before my eyes bled too much. I'll leave
it for the ext4 developers to determine if their code is actually
has this deadlock and how to fix it if it has.
There's a few ways I can see avoid this deadlock. There's probably
more, but these are the first I've though of:
1. get rid of range_cyclic altogether
2. range_cyclic always stops at EOF, and we start again from
writeback index 0 on the next call into write_cache_pages()
2a. wcp also returns EAGAIN to ->writepages implementations to
indicate range cyclic has hit EOF. write-pages implementations can
then flush the current context and call wpc again to continue. i.e.
lift the retry into the ->writepages implementation
Btrfs has a variation of 2a in our bastardized copy of
write_cache_pages(), it has a call to flush the bios we've built up
before doing operations that might deadlock, including waiting for
writeback, locking pages etc:
if (wbc->sync_mode != WB_SYNC_NONE) {
if (PageWriteback(page))
flush_write_bio(epd);
wait_on_page_writeback(page);
}
I don't see any problems with the solution you picked instead, but if we
run into more complex variations of this we can just pass a callback and
a flushing cookie to generic_writepages().
Open question on the perf benefits of staring IO early while we wait for
writeback on page X or letting our bio build bigger.
Open question #2, at least for btrfs we're building a single fat bio by
adding pages to it along the way. This is a small variation on
plugging...we could just pull the last bio off the plug stack and stuff
the pages in. Then we'd get all the deadlock prevent implicit in
plugging for free.
-chris