aboutsummaryrefslogtreecommitdiffstats
path: root/fs/gfs2
diff options
context:
space:
mode:
authorTheodore Ts'o <tytso@mit.edu>2015-02-02 00:37:00 -0500
committerAl Viro <viro@zeniv.linux.org.uk>2015-02-05 02:45:00 -0500
commit0ae45f63d4ef8d8eeec49c7d8b44a1775fff13e8 (patch)
tree660dbb014482092361eab263847fb906b5a9ec22 /fs/gfs2
parente36f014edff70fc02b3d3d79cead1d58f289332e (diff)
vfs: add support for a lazytime mount option
Add a new mount option which enables a new "lazytime" mode. This mode causes atime, mtime, and ctime updates to only be made to the in-memory version of the inode. The on-disk times will only get updated when (a) if the inode needs to be updated for some non-time related change, (b) if userspace calls fsync(), syncfs() or sync(), or (c) just before an undeleted inode is evicted from memory. This is OK according to POSIX because there are no guarantees after a crash unless userspace explicitly requests via a fsync(2) call. For workloads which feature a large number of random write to a preallocated file, the lazytime mount option significantly reduces writes to the inode table. The repeated 4k writes to a single block will result in undesirable stress on flash devices and SMR disk drives. Even on conventional HDD's, the repeated writes to the inode table block will trigger Adjacent Track Interference (ATI) remediation latencies, which very negatively impact long tail latencies --- which is a very big deal for web serving tiers (for example). Google-Bug-Id: 18297052 Signed-off-by: Theodore Ts'o <tytso@mit.edu> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Diffstat (limited to 'fs/gfs2')
-rw-r--r--fs/gfs2/file.c4
1 files changed, 2 insertions, 2 deletions
diff --git a/fs/gfs2/file.c b/fs/gfs2/file.c
index 6e600abf694a..15c44cf457cc 100644
--- a/fs/gfs2/file.c
+++ b/fs/gfs2/file.c
@@ -655,7 +655,7 @@ static int gfs2_fsync(struct file *file, loff_t start, loff_t end,
655{ 655{
656 struct address_space *mapping = file->f_mapping; 656 struct address_space *mapping = file->f_mapping;
657 struct inode *inode = mapping->host; 657 struct inode *inode = mapping->host;
658 int sync_state = inode->i_state & I_DIRTY; 658 int sync_state = inode->i_state & I_DIRTY_ALL;
659 struct gfs2_inode *ip = GFS2_I(inode); 659 struct gfs2_inode *ip = GFS2_I(inode);
660 int ret = 0, ret1 = 0; 660 int ret = 0, ret1 = 0;
661 661
@@ -668,7 +668,7 @@ static int gfs2_fsync(struct file *file, loff_t start, loff_t end,
668 if (!gfs2_is_jdata(ip)) 668 if (!gfs2_is_jdata(ip))
669 sync_state &= ~I_DIRTY_PAGES; 669 sync_state &= ~I_DIRTY_PAGES;
670 if (datasync) 670 if (datasync)
671 sync_state &= ~I_DIRTY_SYNC; 671 sync_state &= ~(I_DIRTY_SYNC | I_DIRTY_TIME);
672 672
673 if (sync_state) { 673 if (sync_state) {
674 ret = sync_inode_metadata(inode, 1); 674 ret = sync_inode_metadata(inode, 1);