Commit 63efe3eb authored by Christoph Hellwig's avatar Christoph Hellwig Committed by Nathan Scott

[XFS] Properly account for clustered pages in the writeout path

SGI Modid: xfs-linux:xfs-kern:171157a
parent dcb5fff8
...@@ -53,10 +53,11 @@ ...@@ -53,10 +53,11 @@
#include "xfs_rw.h" #include "xfs_rw.h"
#include "xfs_iomap.h" #include "xfs_iomap.h"
#include <linux/mpage.h> #include <linux/mpage.h>
#include <linux/writeback.h>
STATIC void xfs_count_page_state(struct page *, int *, int *, int *); STATIC void xfs_count_page_state(struct page *, int *, int *, int *);
STATIC void xfs_convert_page(struct inode *, struct page *, STATIC void xfs_convert_page(struct inode *, struct page *, xfs_iomap_t *,
xfs_iomap_t *, void *, int, int); struct writeback_control *wbc, void *, int, int);
#if defined(XFS_RW_TRACE) #if defined(XFS_RW_TRACE)
void void
...@@ -440,6 +441,7 @@ xfs_map_unwritten( ...@@ -440,6 +441,7 @@ xfs_map_unwritten(
unsigned long p_offset, unsigned long p_offset,
int block_bits, int block_bits,
xfs_iomap_t *iomapp, xfs_iomap_t *iomapp,
struct writeback_control *wbc,
int startio, int startio,
int all_bh) int all_bh)
{ {
...@@ -513,7 +515,7 @@ xfs_map_unwritten( ...@@ -513,7 +515,7 @@ xfs_map_unwritten(
break; break;
nblocks += bs; nblocks += bs;
atomic_add(bs, &pb->pb_io_remaining); atomic_add(bs, &pb->pb_io_remaining);
xfs_convert_page(inode, page, iomapp, pb, xfs_convert_page(inode, page, iomapp, wbc, pb,
startio, all_bh); startio, all_bh);
/* stop if converting the next page might add /* stop if converting the next page might add
* enough blocks that the corresponding byte * enough blocks that the corresponding byte
...@@ -530,7 +532,7 @@ xfs_map_unwritten( ...@@ -530,7 +532,7 @@ xfs_map_unwritten(
if (page) { if (page) {
nblocks += bs; nblocks += bs;
atomic_add(bs, &pb->pb_io_remaining); atomic_add(bs, &pb->pb_io_remaining);
xfs_convert_page(inode, page, iomapp, pb, xfs_convert_page(inode, page, iomapp, wbc, pb,
startio, all_bh); startio, all_bh);
if (nblocks >= ((ULONG_MAX - PAGE_SIZE) >> block_bits)) if (nblocks >= ((ULONG_MAX - PAGE_SIZE) >> block_bits))
goto enough; goto enough;
...@@ -598,6 +600,7 @@ xfs_convert_page( ...@@ -598,6 +600,7 @@ xfs_convert_page(
struct inode *inode, struct inode *inode,
struct page *page, struct page *page,
xfs_iomap_t *iomapp, xfs_iomap_t *iomapp,
struct writeback_control *wbc,
void *private, void *private,
int startio, int startio,
int all_bh) int all_bh)
...@@ -640,8 +643,8 @@ xfs_convert_page( ...@@ -640,8 +643,8 @@ xfs_convert_page(
*/ */
if (buffer_unwritten(bh) && !bh->b_end_io) { if (buffer_unwritten(bh) && !bh->b_end_io) {
ASSERT(tmp->iomap_flags & IOMAP_UNWRITTEN); ASSERT(tmp->iomap_flags & IOMAP_UNWRITTEN);
xfs_map_unwritten(inode, page, head, bh, xfs_map_unwritten(inode, page, head, bh, offset,
offset, bbits, tmp, startio, all_bh); bbits, tmp, wbc, startio, all_bh);
} else if (! (buffer_unwritten(bh) && buffer_locked(bh))) { } else if (! (buffer_unwritten(bh) && buffer_locked(bh))) {
xfs_map_at_offset(page, bh, offset, bbits, tmp); xfs_map_at_offset(page, bh, offset, bbits, tmp);
if (buffer_unwritten(bh)) { if (buffer_unwritten(bh)) {
...@@ -660,6 +663,7 @@ xfs_convert_page( ...@@ -660,6 +663,7 @@ xfs_convert_page(
} while (i++, (bh = bh->b_this_page) != head); } while (i++, (bh = bh->b_this_page) != head);
if (startio) { if (startio) {
wbc->nr_to_write--;
xfs_submit_page(page, bh_arr, index); xfs_submit_page(page, bh_arr, index);
} else { } else {
unlock_page(page); unlock_page(page);
...@@ -675,6 +679,7 @@ xfs_cluster_write( ...@@ -675,6 +679,7 @@ xfs_cluster_write(
struct inode *inode, struct inode *inode,
pgoff_t tindex, pgoff_t tindex,
xfs_iomap_t *iomapp, xfs_iomap_t *iomapp,
struct writeback_control *wbc,
int startio, int startio,
int all_bh) int all_bh)
{ {
...@@ -686,7 +691,8 @@ xfs_cluster_write( ...@@ -686,7 +691,8 @@ xfs_cluster_write(
page = xfs_probe_delalloc_page(inode, tindex); page = xfs_probe_delalloc_page(inode, tindex);
if (!page) if (!page)
break; break;
xfs_convert_page(inode, page, iomapp, NULL, startio, all_bh); xfs_convert_page(inode, page, iomapp, wbc, NULL,
startio, all_bh);
} }
} }
...@@ -713,6 +719,7 @@ STATIC int ...@@ -713,6 +719,7 @@ STATIC int
xfs_page_state_convert( xfs_page_state_convert(
struct inode *inode, struct inode *inode,
struct page *page, struct page *page,
struct writeback_control *wbc,
int startio, int startio,
int unmapped) /* also implies page uptodate */ int unmapped) /* also implies page uptodate */
{ {
...@@ -776,7 +783,7 @@ xfs_page_state_convert( ...@@ -776,7 +783,7 @@ xfs_page_state_convert(
err = xfs_map_unwritten(inode, page, err = xfs_map_unwritten(inode, page,
head, bh, p_offset, head, bh, p_offset,
inode->i_blkbits, iomp, inode->i_blkbits, iomp,
startio, unmapped); wbc, startio, unmapped);
if (err) { if (err) {
goto error; goto error;
} }
...@@ -863,8 +870,10 @@ xfs_page_state_convert( ...@@ -863,8 +870,10 @@ xfs_page_state_convert(
if (startio) if (startio)
xfs_submit_page(page, bh_arr, cnt); xfs_submit_page(page, bh_arr, cnt);
if (iomp) if (iomp) {
xfs_cluster_write(inode, page->index + 1, iomp, startio, unmapped); xfs_cluster_write(inode, page->index + 1, iomp, wbc,
startio, unmapped);
}
return page_dirty; return page_dirty;
...@@ -1173,7 +1182,7 @@ linvfs_writepage( ...@@ -1173,7 +1182,7 @@ linvfs_writepage(
* Convert delayed allocate, unwritten or unmapped space * Convert delayed allocate, unwritten or unmapped space
* to real space and flush out to disk. * to real space and flush out to disk.
*/ */
error = xfs_page_state_convert(inode, page, 1, unmapped); error = xfs_page_state_convert(inode, page, wbc, 1, unmapped);
if (error == -EAGAIN) if (error == -EAGAIN)
goto out_fail; goto out_fail;
if (unlikely(error < 0)) if (unlikely(error < 0))
...@@ -1216,6 +1225,10 @@ linvfs_release_page( ...@@ -1216,6 +1225,10 @@ linvfs_release_page(
{ {
struct inode *inode = page->mapping->host; struct inode *inode = page->mapping->host;
int dirty, delalloc, unmapped, unwritten; int dirty, delalloc, unmapped, unwritten;
struct writeback_control wbc = {
.sync_mode = WB_SYNC_ALL,
.nr_to_write = 1,
};
xfs_page_trace(XFS_RELEASEPAGE_ENTER, inode, page, gfp_mask); xfs_page_trace(XFS_RELEASEPAGE_ENTER, inode, page, gfp_mask);
...@@ -1238,7 +1251,7 @@ linvfs_release_page( ...@@ -1238,7 +1251,7 @@ linvfs_release_page(
* Never need to allocate space here - we will always * Never need to allocate space here - we will always
* come back to writepage in that case. * come back to writepage in that case.
*/ */
dirty = xfs_page_state_convert(inode, page, 0, 0); dirty = xfs_page_state_convert(inode, page, &wbc, 0, 0);
if (dirty == 0 && !unwritten) if (dirty == 0 && !unwritten)
goto free_buffers; goto free_buffers;
return 0; return 0;
......
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