xfs: remove xfs_cancel_ioend

We currently have code to cancel ioends being built because we
change bufferhead state as we build the ioend. On error, this needs
to be unwound and so we have cancelling code that walks the buffers
on the ioend chain and undoes these state changes.

However, the IO submission path already handles state changes for
buffers when a submission error occurs, so we don't really need a
separate cancel function to do this - we can simply submit the
ioend chain with the specific error and it will be cancelled rather
than submitted.

Hence we can remove the explicit cancel code and just rely on
submission to deal with the error correctly.

Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Signed-off-by: Dave Chinner <david@fromorbit.com>
This commit is contained in:
Dave Chinner 2016-02-15 17:21:12 +11:00 committed by Dave Chinner
parent 988ef92792
commit 150d5be09c
1 changed files with 47 additions and 48 deletions

View File

@ -521,38 +521,6 @@ xfs_submit_ioend(
} while ((ioend = next) != NULL);
}
/*
* Cancel submission of all buffer_heads so far in this endio.
* Toss the endio too. Only ever called for the initial page
* in a writepage request, so only ever one page.
*/
STATIC void
xfs_cancel_ioend(
xfs_ioend_t *ioend)
{
xfs_ioend_t *next;
struct buffer_head *bh, *next_bh;
do {
next = ioend->io_list;
bh = ioend->io_buffer_head;
do {
next_bh = bh->b_private;
clear_buffer_async_write(bh);
/*
* The unwritten flag is cleared when added to the
* ioend. We're not submitting for I/O so mark the
* buffer unwritten again for next time around.
*/
if (ioend->io_type == XFS_IO_UNWRITTEN)
set_buffer_unwritten(bh);
unlock_buffer(bh);
} while ((bh = next_bh) != NULL);
mempool_free(ioend, xfs_ioend_pool);
} while ((ioend = next) != NULL);
}
/*
* Test to see if we've been building up a completion structure for
* earlier buffers -- if so, we try to append to this ioend if we
@ -925,6 +893,28 @@ xfs_aops_discard_page(
return;
}
static int
xfs_writepage_submit(
struct xfs_ioend *ioend,
struct xfs_ioend *iohead,
struct writeback_control *wbc,
int status)
{
struct blk_plug plug;
/* Reserve log space if we might write beyond the on-disk inode size. */
if (!status && ioend && ioend->io_type != XFS_IO_UNWRITTEN &&
xfs_ioend_is_append(ioend))
status = xfs_setfilesize_trans_alloc(ioend);
if (iohead) {
blk_start_plug(&plug);
xfs_submit_ioend(wbc, iohead, status);
blk_finish_plug(&plug);
}
return status;
}
/*
* Write out a dirty page.
*
@ -1136,6 +1126,7 @@ xfs_vm_writepage(
return 0;
ASSERT(iohead);
ASSERT(err == 0);
/*
* Any errors from this point onwards need tobe reported through the IO
@ -1161,25 +1152,33 @@ xfs_vm_writepage(
wbc, end_index);
}
/*
* Reserve log space if we might write beyond the on-disk inode size.
*/
err = 0;
if (ioend->io_type != XFS_IO_UNWRITTEN && xfs_ioend_is_append(ioend))
err = xfs_setfilesize_trans_alloc(ioend);
xfs_submit_ioend(wbc, iohead, err);
return 0;
return xfs_writepage_submit(ioend, iohead, wbc, 0);
error:
if (iohead)
xfs_cancel_ioend(iohead);
/*
* On error, we have to fail the iohead here because we buffers locked
* in the ioend chain. If we don't do this, we'll deadlock invalidating
* the page as that tries to lock the buffers on the page. Also, because
* we may have set pages under writeback, we have to run IO completion to
* mark the error state of the IO appropriately, so we can't cancel the
* ioend directly here. That means we have to mark this page as under
* writeback if we included any buffers from it in the ioend chain.
*/
if (count)
xfs_start_page_writeback(page, 0, count);
xfs_writepage_submit(ioend, iohead, wbc, err);
xfs_aops_discard_page(page);
ClearPageUptodate(page);
unlock_page(page);
/*
* We can only discard the page we had the IO error on if we haven't
* included it in the ioend above. If it has already been errored out,
* the it is unlocked and we can't touch it here.
*/
if (!count) {
xfs_aops_discard_page(page);
ClearPageUptodate(page);
unlock_page(page);
}
mapping_set_error(page->mapping, err);
return err;
redirty: