From: Dave Chinner <david@fromorbit.com>
To: Brian Foster <bfoster@redhat.com>
Cc: xfs@oss.sgi.com
Subject: Re: [PATCH] xfs: flush entire last page of old EOF on truncate up
Date: Thu, 18 Sep 2014 07:24:52 +1000 [thread overview]
Message-ID: <20140917212452.GW4322@dastard> (raw)
In-Reply-To: <20140917135711.GB29222@bfoster.bfoster>
On Wed, Sep 17, 2014 at 09:57:11AM -0400, Brian Foster wrote:
> On Wed, Sep 17, 2014 at 08:45:47AM +1000, Dave Chinner wrote:
> > From: Dave Chinner <dchinner@redhat.com>
> >
> > On a sub-page sized filesystem, truncating a mapped region down
> > leaves us in a world of hurt. We truncate the pagecache, zeroing the
> > newly unused tail, then punch blocks out from under the page. If we
> > then truncate the file back up immediately, we expose that unmapped
> > hole to a dirty page mapped into the user application, and that's
> > where it all goes wrong.
> >
> > In truncating the page cache, we avoid unmapping the tail page of
> > the cache because it still contains valid data. The problem is that
> > it also contains a hole after the truncate, but nobody told the mm
> > subsystem that. Therefore, if the page is dirty before the truncate,
> > we'll never get a .page_mkwrite callout after we extend the file and
> > the application writes data into the hole on the page. Hence when
> > we come to writing that region of the page, it has no blocks and no
> > delayed allocation reservation and hence we toss the data away.
> >
> > This patch adds code to the truncate up case to solve it, by
> > ensuring the partial page at the old EOF is always cleaned after we
> > do any zeroing and move the EOF upwards. We can't actually serialise
> > the page writeback and truncate against page faults (yes, that
> > problem AGAIN) so this is really just a best effort and assumes it
> > is extremely unlikely that someone is concurrently writing to the
> > page at the EOF while extending the file.
> >
> > Signed-off-by: Dave Chinner <dchinner@redhat.com>
> > ---
> > fs/xfs/xfs_iops.c | 30 ++++++++++++++++++++++++++++++
> > 1 file changed, 30 insertions(+)
> >
> > diff --git a/fs/xfs/xfs_iops.c b/fs/xfs/xfs_iops.c
> > index 63aeca8..a726d37 100644
> > --- a/fs/xfs/xfs_iops.c
> > +++ b/fs/xfs/xfs_iops.c
> > @@ -853,6 +853,36 @@ xfs_setattr_size(
> > return error;
> > truncate_setsize(inode, newsize);
> >
> > + /*
> > + * The "we can't serialise against page faults" pain gets worse.
> > + *
> > + * If the file is mapped then we have to clean the page at the old EOF
> > + * when extending the file. Extending the file can expose changes the
> > + * underlying page mapping (e.g. from beyond EOF to a hole or
> > + * unwritten), and so on the next attempt to write to that page we need
> > + * to remap it for write. i.e. we need .page_mkwrite() to be called.
> > + * Hence we need to clean the page to clean the pte and so a new write
> > + * fault will be triggered appropriately.
> > + *
> > + * If we do it before we change the inode size, then we can race with a
> > + * page fault that maps the page with exactly the same problem. If we do
> > + * it after we change the file size, then a new page fault can come in
> > + * and allocate space before we've run the rest of the truncate
> > + * transaction. That's kinda grotesque, but it's better than have data
> > + * over a hole, and so that's the lesser evil that has been chosen here.
> > + *
> > + * The real solution, however, is to have some mechanism for locking out
> > + * page faults while a truncate is in progress.
> > + */
> > + if (newsize > oldsize && mapping_mapped(VFS_I(ip)->i_mapping)) {
> > + error = filemap_write_and_wait_range(
> > + VFS_I(ip)->i_mapping,
> > + round_down(oldsize, PAGE_CACHE_SIZE),
> > + round_up(oldsize, PAGE_CACHE_SIZE) - 1);
> > + if (error)
> > + return error;
> > + }
> > +
>
> IIUC, clear_page_dirty_for_io() will clean/writeprotect the page on
> writeback. Hence any future mmap write generates a (soft) fault and
> allows the mm subsystem to invoke mkwrite(), which in turn allows XFS to
> handle the block mapping (e.g., allocating delalloc blocks via
> get_block()) for the mapped write.
That is correct.
> I was starting to think about whether we could combine this with the
> previous writeback in xfs_setattr_size(), but the complex nature of the
> problem, the long comment required and the notion that we might want to
> fix this "for real" some day suggests the the separate hunk is probably
> safer.
I went through several different iterations of flush on truncate
down, combine into xfs_zero_eof, flush on truncate up, flush before
page cache truncate, flush after, etc. They all worked to solve the
problem to some level, but it the deciding factor was that the
truncate down/mwrite to tail page within EOF/truncate up case
requires the flush to be in the truncate up code for mwrite after
truncate up to work correctly. So then it was really a case of
working out where to minimise the potential impact of a racing page
fault.....
> Anyways, if I'm following the above behavior correctly, this
> looks fine to me:
>
> Reviewed-by: Brian Foster <bfoster@redhat.com>
Thanks.
-Dave.
--
Dave Chinner
david@fromorbit.com
_______________________________________________
xfs mailing list
xfs@oss.sgi.com
http://oss.sgi.com/mailman/listinfo/xfs
prev parent reply other threads:[~2014-09-17 21:25 UTC|newest]
Thread overview: 3+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-09-16 22:45 [PATCH] xfs: flush entire last page of old EOF on truncate up Dave Chinner
2014-09-17 13:57 ` Brian Foster
2014-09-17 21:24 ` Dave Chinner [this message]
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20140917212452.GW4322@dastard \
--to=david@fromorbit.com \
--cc=bfoster@redhat.com \
--cc=xfs@oss.sgi.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox