mirror of
https://github.com/FEX-Emu/linux.git
synced 2025-01-08 02:21:18 +00:00
fbcc025613
xfs_vm_writepages() calls generic_writepages to writeback a range of a file, but then xfs_vm_writepage() clusters pages itself as it does not have any context it can pass between->writepage calls from __write_cache_pages(). Introduce a writeback context for xfs_vm_writepages() and call __write_cache_pages directly with our own writepage callback so that we can pass that context to each writepage invocation. This encapsulates the current mapping, whether it is valid or not, the current ioend and it's IO type and the ioend chain being built. This requires us to move the ioend submission up to the level where the writepage context is declared. This does mean we do not submit IO until we packaged the entire writeback range, but with the block plugging in the writepages call this is the way IO is submitted, anyway. It also means that we need to handle discontiguous page ranges. If the pages sent down by write_cache_pages to the writepage callback are discontiguous, we need to detect this and put each discontiguous page range into individual ioends. This is needed to ensure that the ioend accurately represents the range of the file that it covers so that file size updates during IO completion set the size correctly. Failure to take into account the discontiguous ranges results in files being too small when writeback patterns are non-sequential. Signed-off-by: Dave Chinner <dchinner@redhat.com> Reviewed-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Dave Chinner <david@fromorbit.com>
69 lines
2.5 KiB
C
69 lines
2.5 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 xfs_ioend *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 */
|
|
struct buffer_head *io_buffer_head;/* buffer linked list head */
|
|
struct buffer_head *io_buffer_tail;/* buffer linked list tail */
|
|
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 */
|
|
} 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 *);
|
|
|
|
#endif /* __XFS_AOPS_H__ */
|