* Re: [PATCH V4 4/8] Cleancache: VFS hooks for cleancache [not found] <20100830223203.GA1296@ca-server1.us.oracle.com> @ 2010-08-30 22:51 ` Jeremy Fitzhardinge 2010-08-31 0:48 ` Dan Magenheimer 0 siblings, 1 reply; 3+ messages in thread From: Jeremy Fitzhardinge @ 2010-08-30 22:51 UTC (permalink / raw) To: Dan Magenheimer Cc: chris.mason, viro, akpm, adilger, tytso, mfasheh, joel.becker, matthew, linux-btrfs, linux-kernel, linux-fsdevel, linux-ext4, ocfs2-devel, linux-mm, ngupta, JBeulich, kurt.hackel, npiggin, dave.mccracken, riel, avi, konrad.wilk, mel, yinghan, gthelen On 08/30/2010 03:32 PM, Dan Magenheimer wrote: > [PATCH V4 4/8] Cleancache: VFS hooks for cleancache > > Implement core hooks in VFS for: initializing cleancache > per filesystem; capturing clean pages evicted by page cache; > attempting to get pages from cleancache before filesystem > read; and ensuring coherency between pagecache, disk, > and cleancache. All hooks become no-ops if CONFIG_CLEANCACHE > is unset, or become compare-function-pointer-to-NULL if > CONFIG_CLEANCACHE is set but no cleancache "backend" has > claimed cleancache_ops. > > Signed-off-by: Chris Mason <chris.mason@oracle.com> > Signed-off-by: Dan Magenheimer <dan.magenheimer@oracle.com> > > Diffstat: > fs/buffer.c | 5 +++++ > fs/mpage.c | 7 +++++++ > fs/super.c | 7 +++++++ > mm/filemap.c | 11 +++++++++++ > mm/truncate.c | 10 ++++++++++ > 5 files changed, 40 insertions(+) > > --- linux-2.6.36-rc3/fs/super.c 2010-08-29 09:36:04.000000000 -0600 > +++ linux-2.6.36-rc3-cleancache/fs/super.c 2010-08-30 09:20:43.000000000 -0600 > @@ -30,6 +30,7 @@ > #include <linux/idr.h> > #include <linux/mutex.h> > #include <linux/backing-dev.h> > +#include <linux/cleancache.h> > #include "internal.h" > > > @@ -110,6 +111,7 @@ static struct super_block *alloc_super(s > s->s_maxbytes = MAX_NON_LFS; > s->s_op = &default_op; > s->s_time_gran = 1000000000; > + s->cleancache_poolid = -1; > } > out: > return s; > @@ -176,6 +178,11 @@ void deactivate_locked_super(struct supe > struct file_system_type *fs = s->s_type; > if (atomic_dec_and_test(&s->s_active)) { > fs->kill_sb(s); > + if (s->cleancache_poolid >= 0) { > + int cleancache_poolid = s->cleancache_poolid; That's a lot of characters for a local in a 3-line scope. > + s->cleancache_poolid = -1; /* avoid races */ Races with what? Something else sneaking something into the pool after the flush? Is the filesystem dead at this stage or not? > + cleancache_flush_fs(cleancache_poolid); > + } > put_filesystem(fs); > put_super(s); > } else { > --- linux-2.6.36-rc3/fs/buffer.c 2010-08-29 09:36:04.000000000 -0600 > +++ linux-2.6.36-rc3-cleancache/fs/buffer.c 2010-08-30 09:20:42.000000000 -0600 > @@ -41,6 +41,7 @@ > #include <linux/bitops.h> > #include <linux/mpage.h> > #include <linux/bit_spinlock.h> > +#include <linux/cleancache.h> > > static int fsync_buffers_list(spinlock_t *lock, struct list_head *list); > > @@ -277,6 +278,10 @@ void invalidate_bdev(struct block_device > invalidate_bh_lrus(); > lru_add_drain_all(); /* make sure all lru add caches are flushed */ > invalidate_mapping_pages(mapping, 0, -1); > + /* 99% of the time, we don't need to flush the cleancache on the bdev. > + * But, for the strange corners, lets be cautious > + */ This comment-style is... unconventional for the kernel. > + cleancache_flush_inode(mapping); > } > EXPORT_SYMBOL(invalidate_bdev); > > --- linux-2.6.36-rc3/fs/mpage.c 2010-08-29 09:36:04.000000000 -0600 > +++ linux-2.6.36-rc3-cleancache/fs/mpage.c 2010-08-30 09:20:43.000000000 -0600 > @@ -27,6 +27,7 @@ > #include <linux/writeback.h> > #include <linux/backing-dev.h> > #include <linux/pagevec.h> > +#include <linux/cleancache.h> > > /* > * I/O completion handler for multipage BIOs. > @@ -286,6 +287,12 @@ do_mpage_readpage(struct bio *bio, struc > SetPageMappedToDisk(page); > } > > + if (fully_mapped && blocks_per_page == 1 && !PageUptodate(page) && > + cleancache_get_page(page) == 0) { > + SetPageUptodate(page); > + goto confused; > + } > + > /* > * This page will go to BIO. Do we need to send this BIO off first? > */ > --- linux-2.6.36-rc3/mm/filemap.c 2010-08-29 09:36:04.000000000 -0600 > +++ linux-2.6.36-rc3-cleancache/mm/filemap.c 2010-08-30 09:20:43.000000000 -0600 > @@ -34,6 +34,7 @@ > #include <linux/hardirq.h> /* for BUG_ON(!in_atomic()) only */ > #include <linux/memcontrol.h> > #include <linux/mm_inline.h> /* for page_is_file_cache() */ > +#include <linux/cleancache.h> > #include "internal.h" > > /* > @@ -119,6 +120,16 @@ void __remove_from_page_cache(struct pag > { > struct address_space *mapping = page->mapping; > > + /* > + * if we're uptodate, flush out into the cleancache, otherwise > + * invalidate any existing cleancache entries. We can't leave > + * stale data around in the cleancache once our page is gone > + */ > + if (PageUptodate(page)) > + cleancache_put_page(page); > + else > + cleancache_flush_page(mapping, page); > + > radix_tree_delete(&mapping->page_tree, page->index); > page->mapping = NULL; > mapping->nrpages--; > --- linux-2.6.36-rc3/mm/truncate.c 2010-08-29 09:36:04.000000000 -0600 > +++ linux-2.6.36-rc3-cleancache/mm/truncate.c 2010-08-30 09:20:43.000000000 -0600 > @@ -19,6 +19,7 @@ > #include <linux/task_io_accounting_ops.h> > #include <linux/buffer_head.h> /* grr. try_to_release_page, > do_invalidatepage */ > +#include <linux/cleancache.h> > #include "internal.h" > > > @@ -51,6 +52,7 @@ void do_invalidatepage(struct page *page > static inline void truncate_partial_page(struct page *page, unsigned partial) > { > zero_user_segment(page, partial, PAGE_CACHE_SIZE); > + cleancache_flush_page(page->mapping, page); > if (page_has_private(page)) > do_invalidatepage(page, partial); > } > @@ -108,6 +110,10 @@ truncate_complete_page(struct address_sp > clear_page_mlock(page); > remove_from_page_cache(page); > ClearPageMappedToDisk(page); > + /* this must be after the remove_from_page_cache which > + * calls cleancache_put_page (and note page->mapping is now NULL) > + */ > + cleancache_flush_page(mapping, page); > page_cache_release(page); /* pagecache ref */ > return 0; > } > @@ -215,6 +221,7 @@ void truncate_inode_pages_range(struct a > pgoff_t next; > int i; > > + cleancache_flush_inode(mapping); > if (mapping->nrpages == 0) > return; > > @@ -290,6 +297,7 @@ void truncate_inode_pages_range(struct a > pagevec_release(&pvec); > mem_cgroup_uncharge_end(); > } > + cleancache_flush_inode(mapping); > } > EXPORT_SYMBOL(truncate_inode_pages_range); > > @@ -428,6 +436,7 @@ int invalidate_inode_pages2_range(struct > int did_range_unmap = 0; > int wrapped = 0; > > + cleancache_flush_inode(mapping); > pagevec_init(&pvec, 0); > next = start; > while (next <= end && !wrapped && > @@ -486,6 +495,7 @@ int invalidate_inode_pages2_range(struct > mem_cgroup_uncharge_end(); > cond_resched(); > } > + cleancache_flush_inode(mapping); > return ret; > } > EXPORT_SYMBOL_GPL(invalidate_inode_pages2_range); > -- To unsubscribe, send a message with 'unsubscribe linux-mm' in the body to majordomo@kvack.org. For more info on Linux MM, see: http://www.linux-mm.org/ . Don't email: <a href=mailto:"dont@kvack.org"> email@kvack.org </a> ^ permalink raw reply [flat|nested] 3+ messages in thread
* RE: [PATCH V4 4/8] Cleancache: VFS hooks for cleancache 2010-08-30 22:51 ` [PATCH V4 4/8] Cleancache: VFS hooks for cleancache Jeremy Fitzhardinge @ 2010-08-31 0:48 ` Dan Magenheimer 0 siblings, 0 replies; 3+ messages in thread From: Dan Magenheimer @ 2010-08-31 0:48 UTC (permalink / raw) To: Jeremy Fitzhardinge Cc: Chris Mason, viro, akpm, adilger, tytso, mfasheh, Joel Becker, matthew, linux-btrfs, linux-kernel, linux-fsdevel, linux-ext4, ocfs2-devel, linux-mm, ngupta, JBeulich, Kurt Hackel, npiggin, Dave Mccracken, riel, avi, Konrad Wilk, mel, yinghan, gthelen > > + if (s->cleancache_poolid >= 0) { > > + int cleancache_poolid = s->cleancache_poolid; > > That's a lot of characters for a local in a 3-line scope. :-) OK. > > + s->cleancache_poolid = -1; /* avoid races */ > > Races with what? Something else sneaking something into the pool after > the flush? Is the filesystem dead at this stage or not? If there are any inodes in the page cache that point to this superblock, a cleancache_put_page may happen asynchronously that grabs from page->...->i_sb->cleancache_poolid. If the cleancache_flush_fs races with it and another cleancache_init_fs happens after the cleancache_put_page gets the poolid, the poolid could be stale. Highly unlikely, but I thought it best to be safe. > > + /* 99% of the time, we don't need to flush the cleancache on the > bdev. > > + * But, for the strange corners, lets be cautious > > + */ > > This comment-style is... unconventional for the kernel. Yeah, I decided to defer to Chris Mason's wisdom and left it. :-) -- To unsubscribe, send a message with 'unsubscribe linux-mm' in the body to majordomo@kvack.org. For more info on Linux MM, see: http://www.linux-mm.org/ . Don't email: <a href=mailto:"dont@kvack.org"> email@kvack.org </a> ^ permalink raw reply [flat|nested] 3+ messages in thread
* [PATCH V4 4/8] Cleancache: VFS hooks for cleancache
@ 2010-08-30 22:32 Dan Magenheimer
0 siblings, 0 replies; 3+ messages in thread
From: Dan Magenheimer @ 2010-08-30 22:32 UTC (permalink / raw)
To: chris.mason, viro, akpm, adilger, tytso, mfasheh, joel.becker,
matthew, linux-btrfs
[PATCH V4 4/8] Cleancache: VFS hooks for cleancache
Implement core hooks in VFS for: initializing cleancache
per filesystem; capturing clean pages evicted by page cache;
attempting to get pages from cleancache before filesystem
read; and ensuring coherency between pagecache, disk,
and cleancache. All hooks become no-ops if CONFIG_CLEANCACHE
is unset, or become compare-function-pointer-to-NULL if
CONFIG_CLEANCACHE is set but no cleancache "backend" has
claimed cleancache_ops.
Signed-off-by: Chris Mason <chris.mason@oracle.com>
Signed-off-by: Dan Magenheimer <dan.magenheimer@oracle.com>
Diffstat:
fs/buffer.c | 5 +++++
fs/mpage.c | 7 +++++++
fs/super.c | 7 +++++++
mm/filemap.c | 11 +++++++++++
mm/truncate.c | 10 ++++++++++
5 files changed, 40 insertions(+)
--- linux-2.6.36-rc3/fs/super.c 2010-08-29 09:36:04.000000000 -0600
+++ linux-2.6.36-rc3-cleancache/fs/super.c 2010-08-30 09:20:43.000000000 -0600
@@ -30,6 +30,7 @@
#include <linux/idr.h>
#include <linux/mutex.h>
#include <linux/backing-dev.h>
+#include <linux/cleancache.h>
#include "internal.h"
@@ -110,6 +111,7 @@ static struct super_block *alloc_super(s
s->s_maxbytes = MAX_NON_LFS;
s->s_op = &default_op;
s->s_time_gran = 1000000000;
+ s->cleancache_poolid = -1;
}
out:
return s;
@@ -176,6 +178,11 @@ void deactivate_locked_super(struct supe
struct file_system_type *fs = s->s_type;
if (atomic_dec_and_test(&s->s_active)) {
fs->kill_sb(s);
+ if (s->cleancache_poolid >= 0) {
+ int cleancache_poolid = s->cleancache_poolid;
+ s->cleancache_poolid = -1; /* avoid races */
+ cleancache_flush_fs(cleancache_poolid);
+ }
put_filesystem(fs);
put_super(s);
} else {
--- linux-2.6.36-rc3/fs/buffer.c 2010-08-29 09:36:04.000000000 -0600
+++ linux-2.6.36-rc3-cleancache/fs/buffer.c 2010-08-30 09:20:42.000000000 -0600
@@ -41,6 +41,7 @@
#include <linux/bitops.h>
#include <linux/mpage.h>
#include <linux/bit_spinlock.h>
+#include <linux/cleancache.h>
static int fsync_buffers_list(spinlock_t *lock, struct list_head *list);
@@ -277,6 +278,10 @@ void invalidate_bdev(struct block_device
invalidate_bh_lrus();
lru_add_drain_all(); /* make sure all lru add caches are flushed */
invalidate_mapping_pages(mapping, 0, -1);
+ /* 99% of the time, we don't need to flush the cleancache on the bdev.
+ * But, for the strange corners, lets be cautious
+ */
+ cleancache_flush_inode(mapping);
}
EXPORT_SYMBOL(invalidate_bdev);
--- linux-2.6.36-rc3/fs/mpage.c 2010-08-29 09:36:04.000000000 -0600
+++ linux-2.6.36-rc3-cleancache/fs/mpage.c 2010-08-30 09:20:43.000000000 -0600
@@ -27,6 +27,7 @@
#include <linux/writeback.h>
#include <linux/backing-dev.h>
#include <linux/pagevec.h>
+#include <linux/cleancache.h>
/*
* I/O completion handler for multipage BIOs.
@@ -286,6 +287,12 @@ do_mpage_readpage(struct bio *bio, struc
SetPageMappedToDisk(page);
}
+ if (fully_mapped && blocks_per_page == 1 && !PageUptodate(page) &&
+ cleancache_get_page(page) == 0) {
+ SetPageUptodate(page);
+ goto confused;
+ }
+
/*
* This page will go to BIO. Do we need to send this BIO off first?
*/
--- linux-2.6.36-rc3/mm/filemap.c 2010-08-29 09:36:04.000000000 -0600
+++ linux-2.6.36-rc3-cleancache/mm/filemap.c 2010-08-30 09:20:43.000000000 -0600
@@ -34,6 +34,7 @@
#include <linux/hardirq.h> /* for BUG_ON(!in_atomic()) only */
#include <linux/memcontrol.h>
#include <linux/mm_inline.h> /* for page_is_file_cache() */
+#include <linux/cleancache.h>
#include "internal.h"
/*
@@ -119,6 +120,16 @@ void __remove_from_page_cache(struct pag
{
struct address_space *mapping = page->mapping;
+ /*
+ * if we're uptodate, flush out into the cleancache, otherwise
+ * invalidate any existing cleancache entries. We can't leave
+ * stale data around in the cleancache once our page is gone
+ */
+ if (PageUptodate(page))
+ cleancache_put_page(page);
+ else
+ cleancache_flush_page(mapping, page);
+
radix_tree_delete(&mapping->page_tree, page->index);
page->mapping = NULL;
mapping->nrpages--;
--- linux-2.6.36-rc3/mm/truncate.c 2010-08-29 09:36:04.000000000 -0600
+++ linux-2.6.36-rc3-cleancache/mm/truncate.c 2010-08-30 09:20:43.000000000 -0600
@@ -19,6 +19,7 @@
#include <linux/task_io_accounting_ops.h>
#include <linux/buffer_head.h> /* grr. try_to_release_page,
do_invalidatepage */
+#include <linux/cleancache.h>
#include "internal.h"
@@ -51,6 +52,7 @@ void do_invalidatepage(struct page *page
static inline void truncate_partial_page(struct page *page, unsigned partial)
{
zero_user_segment(page, partial, PAGE_CACHE_SIZE);
+ cleancache_flush_page(page->mapping, page);
if (page_has_private(page))
do_invalidatepage(page, partial);
}
@@ -108,6 +110,10 @@ truncate_complete_page(struct address_sp
clear_page_mlock(page);
remove_from_page_cache(page);
ClearPageMappedToDisk(page);
+ /* this must be after the remove_from_page_cache which
+ * calls cleancache_put_page (and note page->mapping is now NULL)
+ */
+ cleancache_flush_page(mapping, page);
page_cache_release(page); /* pagecache ref */
return 0;
}
@@ -215,6 +221,7 @@ void truncate_inode_pages_range(struct a
pgoff_t next;
int i;
+ cleancache_flush_inode(mapping);
if (mapping->nrpages == 0)
return;
@@ -290,6 +297,7 @@ void truncate_inode_pages_range(struct a
pagevec_release(&pvec);
mem_cgroup_uncharge_end();
}
+ cleancache_flush_inode(mapping);
}
EXPORT_SYMBOL(truncate_inode_pages_range);
@@ -428,6 +436,7 @@ int invalidate_inode_pages2_range(struct
int did_range_unmap = 0;
int wrapped = 0;
+ cleancache_flush_inode(mapping);
pagevec_init(&pvec, 0);
next = start;
while (next <= end && !wrapped &&
@@ -486,6 +495,7 @@ int invalidate_inode_pages2_range(struct
mem_cgroup_uncharge_end();
cond_resched();
}
+ cleancache_flush_inode(mapping);
return ret;
}
EXPORT_SYMBOL_GPL(invalidate_inode_pages2_range);
^ permalink raw reply [flat|nested] 3+ messages in threadend of thread, other threads:[~2010-08-31 0:48 UTC | newest]
Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
[not found] <20100830223203.GA1296@ca-server1.us.oracle.com>
2010-08-30 22:51 ` [PATCH V4 4/8] Cleancache: VFS hooks for cleancache Jeremy Fitzhardinge
2010-08-31 0:48 ` Dan Magenheimer
2010-08-30 22:32 Dan Magenheimer
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox