From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: with ECARTIS (v1.0.0; list xfs); Sun, 14 Sep 2008 20:12:33 -0700 (PDT) Received: from relay.sgi.com (relay1.corp.sgi.com [192.26.58.214]) by oss.sgi.com (8.12.11.20060308/8.12.11/SuSE Linux 0.7) with ESMTP id m8F3CVU5002585 for ; Sun, 14 Sep 2008 20:12:31 -0700 Message-ID: <48CDD4EE.8040105@sgi.com> Date: Mon, 15 Sep 2008 13:22:22 +1000 From: Lachlan McIlroy Reply-To: lachlan@sgi.com MIME-Version: 1.0 Subject: Re: [PATCH] Re-dirty pages on I/O error References: <48C8D8CD.7050508@sgi.com> <20080913041930.GC5811@disturbed> In-Reply-To: <20080913041930.GC5811@disturbed> Content-Type: text/plain; charset=ISO-8859-1; format=flowed Content-Transfer-Encoding: 7bit Sender: xfs-bounce@oss.sgi.com Errors-to: xfs-bounce@oss.sgi.com List-Id: xfs To: Lachlan McIlroy , xfs-dev , xfs-oss Dave Chinner wrote: > On Thu, Sep 11, 2008 at 06:37:33PM +1000, Lachlan McIlroy wrote: >> If we get an error in xfs_page_state_convert() - and it's not EAGAIN - then >> we throw away the dirty page without converting the delayed allocation. This >> leaves delayed allocations that can never be removed and confuses code that >> expects a flush of the file to clear them. We need to re-dirty the page on >> error so we can try again later or report that the flush failed. >> >> --- a/fs/xfs/linux-2.6/xfs_aops.c 2008-09-11 16:32:11.000000000 +1000 >> +++ b/fs/xfs/linux-2.6/xfs_aops.c 2008-09-11 15:44:09.000000000 +1000 >> @@ -1147,16 +1147,6 @@ error: >> if (iohead) >> xfs_cancel_ioend(iohead); >> >> - /* >> - * If it's delalloc and we have nowhere to put it, >> - * throw it away, unless the lower layers told >> - * us to try again. >> - */ >> - if (err != -EAGAIN) { >> - if (!unmapped) >> - block_invalidatepage(page, 0); >> - ClearPageUptodate(page); >> - } >> return err; >> } > > So we keep dirty pages around that we can't write back? Yes. > If we are in a low memory situation and the block device > has gone bad, that will prevent memory reclaim from making > progress. How do you differentiate "gone bad" from temporarily unavailable? > > i.e. if we have a bad disk, a user can now take down the system > by running it out of clean memory.... I'm sure there's many ways a malicious user could already do that. Would you rather have data corruption? We've allowed the write() to succeed. We've accepted the data. We have an obligation to write it do disk. Either we keep trying in the face of errors or we take down the filesystem. > > The EAGAIN case is for when we can't get the locks we > need during non-blocking writeback, which is a common case if > there is concurrent writes to this inode.... > >> @@ -1216,8 +1206,11 @@ xfs_vm_writepage( >> * then mark the page dirty again and leave the page >> * as is. >> */ >> - if (current_test_flags(PF_FSTRANS) && need_trans) >> - goto out_fail; >> + if (current_test_flags(PF_FSTRANS) && need_trans) { >> + redirty_page_for_writepage(wbc, page); >> + unlock_page(page); >> + return -EAGAIN; >> + } > > Should not return an error here - the redirty_page_for_writepage() > call effective says "can't do this right away, but no error > needs to be reported because it can be written again later". > Happens all the time with non-blocking writes. Christoph already pointed that one out. > >> @@ -1231,20 +1224,14 @@ xfs_vm_writepage( >> * to real space and flush out to disk. >> */ >> error = xfs_page_state_convert(inode, page, wbc, 1, unmapped); >> - if (error == -EAGAIN) >> - goto out_fail; >> - if (unlikely(error < 0)) >> - goto out_unlock; >> >> - return 0; >> + if (error < 0) { >> + redirty_page_for_writepage(wbc, page); >> + unlock_page(page); >> + return error; >> + } > > That needs the EAGAIN exception - that's not an error. > For EIO, though, we should really be invalidating the > page like we currently do. However, it should be silent as > per the current behaviour - a rate-limited log warning is > really needed here... The EAGAIN case can be exceptioned. The error we are getting here is ENOSPC because xfs_trans_reserve() is failing. It looks to me like __writepage() and mapping_set_error() want to know about that error. We also need that error to be propogated back to any callers of xfs_flushinval_pages() and the only way to do that is to return the actual error. Just redirtying the page wont return an error all the way back up the call stack. Silently discarding data just doesn't make sense. Issuing a log message isn't much better - noone will look for it until after they realise there's a problem and all their files are corrupt.