Commit 150d5be0 authored by Dave Chinner's avatar Dave Chinner Committed by Dave Chinner

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: default avatarDave Chinner <dchinner@redhat.com>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
Signed-off-by: default avatarDave Chinner <david@fromorbit.com>
parent 988ef927
...@@ -521,38 +521,6 @@ xfs_submit_ioend( ...@@ -521,38 +521,6 @@ xfs_submit_ioend(
} while ((ioend = next) != NULL); } 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 * 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 * earlier buffers -- if so, we try to append to this ioend if we
...@@ -925,6 +893,28 @@ xfs_aops_discard_page( ...@@ -925,6 +893,28 @@ xfs_aops_discard_page(
return; 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. * Write out a dirty page.
* *
...@@ -1136,6 +1126,7 @@ xfs_vm_writepage( ...@@ -1136,6 +1126,7 @@ xfs_vm_writepage(
return 0; return 0;
ASSERT(iohead); ASSERT(iohead);
ASSERT(err == 0);
/* /*
* Any errors from this point onwards need tobe reported through the IO * Any errors from this point onwards need tobe reported through the IO
...@@ -1161,25 +1152,33 @@ xfs_vm_writepage( ...@@ -1161,25 +1152,33 @@ xfs_vm_writepage(
wbc, end_index); wbc, end_index);
} }
return xfs_writepage_submit(ioend, iohead, wbc, 0);
error:
/* /*
* Reserve log space if we might write beyond the on-disk inode size. * 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.
*/ */
err = 0; if (count)
if (ioend->io_type != XFS_IO_UNWRITTEN && xfs_ioend_is_append(ioend)) xfs_start_page_writeback(page, 0, count);
err = xfs_setfilesize_trans_alloc(ioend); xfs_writepage_submit(ioend, iohead, wbc, err);
xfs_submit_ioend(wbc, iohead, err);
return 0;
error:
if (iohead)
xfs_cancel_ioend(iohead);
/*
* 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); xfs_aops_discard_page(page);
ClearPageUptodate(page); ClearPageUptodate(page);
unlock_page(page); unlock_page(page);
}
mapping_set_error(page->mapping, err);
return err; return err;
redirty: redirty:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment