
Completion of an ioend requires us to walk the bufferhead list to end writback on all the bufferheads. This, in turn, is needed so that we can end writeback on all the pages we just did IO on. To remove our dependency on bufferheads in writeback, we need to turn this around the other way - we need to walk the pages we've just completed IO on, and then walk the buffers attached to the pages and complete their IO. In doing this, we remove the requirement for the ioend to track bufferheads directly. To enable IO completion to walk all the pages we've submitted IO on, we need to keep the bios that we used for IO around until the ioend has been completed. We can do this simply by chaining the bios to the ioend at completion time, and then walking their pages directly just before destroying the ioend. Signed-off-by: Dave Chinner <dchinner@redhat.com> [hch: changed the xfs_finish_page_writeback calling convention] Signed-off-by: Christoph Hellwig <hch@lst.de> Reviewed-by: Brian Foster <bfoster@redhat.com> Signed-off-by: Dave Chinner <david@fromorbit.com>
72 lines
2.6 KiB
C
72 lines
2.6 KiB
C
/*
|
|
* Copyright (c) 2005-2006 Silicon Graphics, Inc.
|
|
* All Rights Reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it would be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
#ifndef __XFS_AOPS_H__
|
|
#define __XFS_AOPS_H__
|
|
|
|
extern mempool_t *xfs_ioend_pool;
|
|
|
|
/*
|
|
* Types of I/O for bmap clustering and I/O completion tracking.
|
|
*/
|
|
enum {
|
|
XFS_IO_INVALID, /* initial state */
|
|
XFS_IO_DELALLOC, /* covers delalloc region */
|
|
XFS_IO_UNWRITTEN, /* covers allocated but uninitialized data */
|
|
XFS_IO_OVERWRITE, /* covers already allocated extent */
|
|
};
|
|
|
|
#define XFS_IO_TYPES \
|
|
{ XFS_IO_INVALID, "invalid" }, \
|
|
{ XFS_IO_DELALLOC, "delalloc" }, \
|
|
{ XFS_IO_UNWRITTEN, "unwritten" }, \
|
|
{ XFS_IO_OVERWRITE, "overwrite" }
|
|
|
|
/*
|
|
* xfs_ioend struct manages large extent writes for XFS.
|
|
* It can manage several multi-page bio's at once.
|
|
*/
|
|
typedef struct xfs_ioend {
|
|
struct list_head io_list; /* next ioend in chain */
|
|
unsigned int io_type; /* delalloc / unwritten */
|
|
int io_error; /* I/O error code */
|
|
atomic_t io_remaining; /* hold count */
|
|
struct inode *io_inode; /* file being written to */
|
|
size_t io_size; /* size of the extent */
|
|
xfs_off_t io_offset; /* offset in the file */
|
|
struct work_struct io_work; /* xfsdatad work queue */
|
|
struct xfs_trans *io_append_trans;/* xact. for size update */
|
|
struct bio *io_bio; /* bio being built */
|
|
struct bio *io_bio_done; /* bios completed */
|
|
struct bio *io_bio_done_tail; /* bios completed */
|
|
spinlock_t io_lock; /* for bio completion list */
|
|
} xfs_ioend_t;
|
|
|
|
extern const struct address_space_operations xfs_address_space_operations;
|
|
|
|
int xfs_get_blocks(struct inode *inode, sector_t offset,
|
|
struct buffer_head *map_bh, int create);
|
|
int xfs_get_blocks_direct(struct inode *inode, sector_t offset,
|
|
struct buffer_head *map_bh, int create);
|
|
int xfs_get_blocks_dax_fault(struct inode *inode, sector_t offset,
|
|
struct buffer_head *map_bh, int create);
|
|
|
|
extern void xfs_count_page_state(struct page *, int *, int *);
|
|
extern struct block_device *xfs_find_bdev_for_inode(struct inode *);
|
|
|
|
#endif /* __XFS_AOPS_H__ */
|