|
@@ -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>
|
|
@@ -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
|
|
|
|
|
|
@@ -220,24 +222,17 @@ 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);
|
|
}
|
|
}
|
|
|
|
|
|
-/*
|
|
|
|
- * 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)
|
|
|
|
|
|
+static struct nfs_page *nfs_find_and_lock_request(struct page *page)
|
|
{
|
|
{
|
|
struct inode *inode = page->mapping->host;
|
|
struct inode *inode = page->mapping->host;
|
|
struct nfs_page *req;
|
|
struct nfs_page *req;
|
|
int ret;
|
|
int ret;
|
|
|
|
|
|
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) {
|
|
|
|
- spin_unlock(&inode->i_lock);
|
|
|
|
- return 0;
|
|
|
|
- }
|
|
|
|
|
|
+ if (req == NULL)
|
|
|
|
+ break;
|
|
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,
|
|
@@ -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);
|
|
|
|
- BUG();
|
|
|
|
- }
|
|
|
|
spin_unlock(&inode->i_lock);
|
|
spin_unlock(&inode->i_lock);
|
|
|
|
+ return req;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/*
|
|
|
|
+ * 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)
|
|
@@ -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",
|