Loading fs/nfs/file.c +1 −0 Original line number Original line Diff line number Diff line Loading @@ -479,6 +479,7 @@ const struct address_space_operations nfs_file_aops = { .invalidatepage = nfs_invalidate_page, .invalidatepage = nfs_invalidate_page, .releasepage = nfs_release_page, .releasepage = nfs_release_page, .direct_IO = nfs_direct_IO, .direct_IO = nfs_direct_IO, .migratepage = nfs_migrate_page, .launder_page = nfs_launder_page, .launder_page = nfs_launder_page, }; }; Loading fs/nfs/internal.h +6 −0 Original line number Original line Diff line number Diff line Loading @@ -248,6 +248,12 @@ extern void nfs_read_prepare(struct rpc_task *task, void *calldata); /* write.c */ /* write.c */ extern void nfs_write_prepare(struct rpc_task *task, void *calldata); extern void nfs_write_prepare(struct rpc_task *task, void *calldata); #ifdef CONFIG_MIGRATION extern int nfs_migrate_page(struct address_space *, struct page *, struct page *); #else #define nfs_migrate_page NULL #endif /* nfs4proc.c */ /* nfs4proc.c */ extern int _nfs4_call_sync(struct nfs_server *server, extern int _nfs4_call_sync(struct nfs_server *server, Loading fs/nfs/write.c +69 −22 Original line number Original line Diff line number Diff line Loading @@ -13,6 +13,7 @@ #include <linux/file.h> #include <linux/file.h> #include <linux/writeback.h> #include <linux/writeback.h> #include <linux/swap.h> #include <linux/swap.h> #include <linux/migrate.h> #include <linux/sunrpc/clnt.h> #include <linux/sunrpc/clnt.h> #include <linux/nfs_fs.h> #include <linux/nfs_fs.h> Loading @@ -26,6 +27,7 @@ #include "internal.h" #include "internal.h" #include "iostat.h" #include "iostat.h" #include "nfs4_fs.h" #include "nfs4_fs.h" #include "fscache.h" #define NFSDBG_FACILITY NFSDBG_PAGECACHE #define NFSDBG_FACILITY NFSDBG_PAGECACHE Loading Loading @@ -220,12 +222,7 @@ static void nfs_end_page_writeback(struct page *page) clear_bdi_congested(&nfss->backing_dev_info, BLK_RW_ASYNC); clear_bdi_congested(&nfss->backing_dev_info, BLK_RW_ASYNC); } } /* static struct nfs_page *nfs_find_and_lock_request(struct page *page) * Find an associated nfs write request, and prepare to flush it out * May return an error if the user signalled nfs_wait_on_request(). */ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, struct page *page) { { struct inode *inode = page->mapping->host; struct inode *inode = page->mapping->host; struct nfs_page *req; struct nfs_page *req; Loading @@ -234,10 +231,8 @@ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, spin_lock(&inode->i_lock); spin_lock(&inode->i_lock); for (;;) { for (;;) { req = nfs_page_find_request_locked(page); req = nfs_page_find_request_locked(page); if (req == NULL) { if (req == NULL) spin_unlock(&inode->i_lock); break; return 0; } if (nfs_set_page_tag_locked(req)) if (nfs_set_page_tag_locked(req)) break; break; /* Note: If we hold the page lock, as is the case in nfs_writepage, /* Note: If we hold the page lock, as is the case in nfs_writepage, Loading @@ -249,23 +244,40 @@ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, ret = nfs_wait_on_request(req); ret = nfs_wait_on_request(req); nfs_release_request(req); nfs_release_request(req); if (ret != 0) if (ret != 0) return ret; return ERR_PTR(ret); spin_lock(&inode->i_lock); spin_lock(&inode->i_lock); } } if (test_bit(PG_CLEAN, &req->wb_flags)) { spin_unlock(&inode->i_lock); BUG(); } if (nfs_set_page_writeback(page) != 0) { spin_unlock(&inode->i_lock); spin_unlock(&inode->i_lock); BUG(); return req; } } spin_unlock(&inode->i_lock); /* * Find an associated nfs write request, and prepare to flush it out * May return an error if the user signalled nfs_wait_on_request(). */ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, struct page *page) { struct nfs_page *req; int ret = 0; req = nfs_find_and_lock_request(page); if (!req) goto out; ret = PTR_ERR(req); if (IS_ERR(req)) goto out; ret = nfs_set_page_writeback(page); BUG_ON(ret != 0); BUG_ON(test_bit(PG_CLEAN, &req->wb_flags)); if (!nfs_pageio_add_request(pgio, req)) { if (!nfs_pageio_add_request(pgio, req)) { nfs_redirty_request(req); nfs_redirty_request(req); return pgio->pg_error; ret = pgio->pg_error; } } return 0; out: return ret; } } static int nfs_do_writepage(struct page *page, struct writeback_control *wbc, struct nfs_pageio_descriptor *pgio) static int nfs_do_writepage(struct page *page, struct writeback_control *wbc, struct nfs_pageio_descriptor *pgio) Loading Loading @@ -1582,6 +1594,41 @@ int nfs_wb_page(struct inode *inode, struct page* page) return nfs_wb_page_priority(inode, page, FLUSH_STABLE); return nfs_wb_page_priority(inode, page, FLUSH_STABLE); } } #ifdef CONFIG_MIGRATION int nfs_migrate_page(struct address_space *mapping, struct page *newpage, struct page *page) { struct nfs_page *req; int ret; if (PageFsCache(page)) nfs_fscache_release_page(page, GFP_KERNEL); req = nfs_find_and_lock_request(page); ret = PTR_ERR(req); if (IS_ERR(req)) goto out; ret = migrate_page(mapping, newpage, page); if (!req) goto out; if (ret) goto out_unlock; page_cache_get(newpage); req->wb_page = newpage; SetPagePrivate(newpage); set_page_private(newpage, page_private(page)); ClearPagePrivate(page); set_page_private(page, 0); page_cache_release(page); out_unlock: nfs_clear_page_tag_locked(req); nfs_release_request(req); out: return ret; } #endif int __init nfs_init_writepagecache(void) int __init nfs_init_writepagecache(void) { { nfs_wdata_cachep = kmem_cache_create("nfs_write_data", nfs_wdata_cachep = kmem_cache_create("nfs_write_data", Loading Loading
fs/nfs/file.c +1 −0 Original line number Original line Diff line number Diff line Loading @@ -479,6 +479,7 @@ const struct address_space_operations nfs_file_aops = { .invalidatepage = nfs_invalidate_page, .invalidatepage = nfs_invalidate_page, .releasepage = nfs_release_page, .releasepage = nfs_release_page, .direct_IO = nfs_direct_IO, .direct_IO = nfs_direct_IO, .migratepage = nfs_migrate_page, .launder_page = nfs_launder_page, .launder_page = nfs_launder_page, }; }; Loading
fs/nfs/internal.h +6 −0 Original line number Original line Diff line number Diff line Loading @@ -248,6 +248,12 @@ extern void nfs_read_prepare(struct rpc_task *task, void *calldata); /* write.c */ /* write.c */ extern void nfs_write_prepare(struct rpc_task *task, void *calldata); extern void nfs_write_prepare(struct rpc_task *task, void *calldata); #ifdef CONFIG_MIGRATION extern int nfs_migrate_page(struct address_space *, struct page *, struct page *); #else #define nfs_migrate_page NULL #endif /* nfs4proc.c */ /* nfs4proc.c */ extern int _nfs4_call_sync(struct nfs_server *server, extern int _nfs4_call_sync(struct nfs_server *server, Loading
fs/nfs/write.c +69 −22 Original line number Original line Diff line number Diff line Loading @@ -13,6 +13,7 @@ #include <linux/file.h> #include <linux/file.h> #include <linux/writeback.h> #include <linux/writeback.h> #include <linux/swap.h> #include <linux/swap.h> #include <linux/migrate.h> #include <linux/sunrpc/clnt.h> #include <linux/sunrpc/clnt.h> #include <linux/nfs_fs.h> #include <linux/nfs_fs.h> Loading @@ -26,6 +27,7 @@ #include "internal.h" #include "internal.h" #include "iostat.h" #include "iostat.h" #include "nfs4_fs.h" #include "nfs4_fs.h" #include "fscache.h" #define NFSDBG_FACILITY NFSDBG_PAGECACHE #define NFSDBG_FACILITY NFSDBG_PAGECACHE Loading Loading @@ -220,12 +222,7 @@ static void nfs_end_page_writeback(struct page *page) clear_bdi_congested(&nfss->backing_dev_info, BLK_RW_ASYNC); clear_bdi_congested(&nfss->backing_dev_info, BLK_RW_ASYNC); } } /* static struct nfs_page *nfs_find_and_lock_request(struct page *page) * Find an associated nfs write request, and prepare to flush it out * May return an error if the user signalled nfs_wait_on_request(). */ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, struct page *page) { { struct inode *inode = page->mapping->host; struct inode *inode = page->mapping->host; struct nfs_page *req; struct nfs_page *req; Loading @@ -234,10 +231,8 @@ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, spin_lock(&inode->i_lock); spin_lock(&inode->i_lock); for (;;) { for (;;) { req = nfs_page_find_request_locked(page); req = nfs_page_find_request_locked(page); if (req == NULL) { if (req == NULL) spin_unlock(&inode->i_lock); break; return 0; } if (nfs_set_page_tag_locked(req)) if (nfs_set_page_tag_locked(req)) break; break; /* Note: If we hold the page lock, as is the case in nfs_writepage, /* Note: If we hold the page lock, as is the case in nfs_writepage, Loading @@ -249,23 +244,40 @@ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, ret = nfs_wait_on_request(req); ret = nfs_wait_on_request(req); nfs_release_request(req); nfs_release_request(req); if (ret != 0) if (ret != 0) return ret; return ERR_PTR(ret); spin_lock(&inode->i_lock); spin_lock(&inode->i_lock); } } if (test_bit(PG_CLEAN, &req->wb_flags)) { spin_unlock(&inode->i_lock); BUG(); } if (nfs_set_page_writeback(page) != 0) { spin_unlock(&inode->i_lock); spin_unlock(&inode->i_lock); BUG(); return req; } } spin_unlock(&inode->i_lock); /* * Find an associated nfs write request, and prepare to flush it out * May return an error if the user signalled nfs_wait_on_request(). */ static int nfs_page_async_flush(struct nfs_pageio_descriptor *pgio, struct page *page) { struct nfs_page *req; int ret = 0; req = nfs_find_and_lock_request(page); if (!req) goto out; ret = PTR_ERR(req); if (IS_ERR(req)) goto out; ret = nfs_set_page_writeback(page); BUG_ON(ret != 0); BUG_ON(test_bit(PG_CLEAN, &req->wb_flags)); if (!nfs_pageio_add_request(pgio, req)) { if (!nfs_pageio_add_request(pgio, req)) { nfs_redirty_request(req); nfs_redirty_request(req); return pgio->pg_error; ret = pgio->pg_error; } } return 0; out: return ret; } } static int nfs_do_writepage(struct page *page, struct writeback_control *wbc, struct nfs_pageio_descriptor *pgio) static int nfs_do_writepage(struct page *page, struct writeback_control *wbc, struct nfs_pageio_descriptor *pgio) Loading Loading @@ -1582,6 +1594,41 @@ int nfs_wb_page(struct inode *inode, struct page* page) return nfs_wb_page_priority(inode, page, FLUSH_STABLE); return nfs_wb_page_priority(inode, page, FLUSH_STABLE); } } #ifdef CONFIG_MIGRATION int nfs_migrate_page(struct address_space *mapping, struct page *newpage, struct page *page) { struct nfs_page *req; int ret; if (PageFsCache(page)) nfs_fscache_release_page(page, GFP_KERNEL); req = nfs_find_and_lock_request(page); ret = PTR_ERR(req); if (IS_ERR(req)) goto out; ret = migrate_page(mapping, newpage, page); if (!req) goto out; if (ret) goto out_unlock; page_cache_get(newpage); req->wb_page = newpage; SetPagePrivate(newpage); set_page_private(newpage, page_private(page)); ClearPagePrivate(page); set_page_private(page, 0); page_cache_release(page); out_unlock: nfs_clear_page_tag_locked(req); nfs_release_request(req); out: return ret; } #endif int __init nfs_init_writepagecache(void) int __init nfs_init_writepagecache(void) { { nfs_wdata_cachep = kmem_cache_create("nfs_write_data", nfs_wdata_cachep = kmem_cache_create("nfs_write_data", Loading