diff mbox

[PATCH-v2,1/2] fs: make sure the timestamps for lazytime inodes eventually get written

Message ID 1426533260-3305-2-git-send-email-tytso@mit.edu (mailing list archive)
State New, archived
Headers show

Commit Message

Theodore Ts'o March 16, 2015, 7:14 p.m. UTC
Jan Kara pointed out that if there is an inode which is constantly
getting dirtied with I_DIRTY_PAGES, an inode with an updated timestamp
will never be written since inode->dirtied_when is constantly getting
updated.  We fix this by adding an extra field to the inode,
dirtied_time_when, so inodes with a stale dirtytime can get detected
and handled.

In addition, if we have a dirtytime inode caused by an atime update,
and there is no write activity on the file system, we need to have a
secondary system to make sure these inodes get written out.  We do
this by setting up a second delayed work structure which wakes up the
CPU much more rarely compared to writeback_expire_centisecs.

Signed-off-by: Theodore Ts'o <tytso@mit.edu>
Cc: stable@vger.kernel.org
---
 fs/fs-writeback.c  | 82 +++++++++++++++++++++++++++++++++++++++++++++++-------
 include/linux/fs.h |  1 +
 2 files changed, 73 insertions(+), 10 deletions(-)

Comments

Andreas Dilger March 16, 2015, 9:34 p.m. UTC | #1
On Mar 16, 2015, at 1:14 PM, Theodore Ts'o <tytso@mit.edu> wrote:
> 
> Jan Kara pointed out that if there is an inode which is constantly
> getting dirtied with I_DIRTY_PAGES, an inode with an updated timestamp
> will never be written since inode->dirtied_when is constantly getting
> updated.  We fix this by adding an extra field to the inode,
> dirtied_time_when, so inodes with a stale dirtytime can get detected
> and handled.

The drawback here is that this adds another 8 bytes to every inode for
a field of marginal value, since this is only important for the rare
case of a file that is being dirtied continuously.

I wonder if something more lightweight could be added to avoid this
problem?  For example, we only care about this case if it has been
going on for more than the lazytime interval (about a day), so the
inode could store a 16-bit i_dirtied_time_when that is approximately
(jiffies >> bits_in_a_half_a_day) and only check time_after() that.
The __u16 could fit into some existing hole (e.g. after i_bytes on my
kernel) and avoid expanding the size of the inode at all.

The remaining high bits of i_dirtied_time_when would be irrelevant, since
a __u16 of half-days is about 80 years, so it would be enough to compare:


    time_after(i_dirtied_time_when, (__u16)(jiffies >> bits_in_half_a_day))


A day is 86400s, so 43200s is close to (1 << 22) jiffies for HZ=100, and
(1 << 25) jiffies is about 3/8 of a day for HZ=1000.  Since the exact
times for inode writeout don't matter very much here, having only shifts
to convert jiffies to i_dirtied_time_when in the kernel is better I think.

Minor issue, is there a good reason why dirtied_time_when doesn't have an
"i_" prefix?

Cheers, Andreas

> In addition, if we have a dirtytime inode caused by an atime update,
> and there is no write activity on the file system, we need to have a
> secondary system to make sure these inodes get written out.  We do
> this by setting up a second delayed work structure which wakes up the
> CPU much more rarely compared to writeback_expire_centisecs.
> 
> Signed-off-by: Theodore Ts'o <tytso@mit.edu>
> Cc: stable@vger.kernel.org
> ---
> fs/fs-writeback.c  | 82 +++++++++++++++++++++++++++++++++++++++++++++++-------
> include/linux/fs.h |  1 +
> 2 files changed, 73 insertions(+), 10 deletions(-)
> 
> diff --git a/fs/fs-writeback.c b/fs/fs-writeback.c
> index e907052..ae13fba 100644
> --- a/fs/fs-writeback.c
> +++ b/fs/fs-writeback.c
> @@ -53,6 +53,18 @@ struct wb_writeback_work {
> 	struct completion *done;	/* set if the caller waits */
> };
> 
> +/*
> + * If an inode is constantly having its pages dirtied, but then the
> + * updates stop dirtytime_expire_interval seconds in the past, it's
> + * possible for the worst case time between when an inode has its
> + * timestamps updated and when they finally get written out to be two
> + * dirtytime_expire_intervals.  We set the default to 12 hours (in
> + * seconds), which means most of the time inodes will have their
> + * timestamps written to disk after 12 hours, but in the worst case a
> + * few inodes might not their timestamps updated for 24 hours.
> + */
> +unsigned int dirtytime_expire_interval = 12 * 60 * 60;
> +
> /**
>  * writeback_in_progress - determine whether there is writeback in progress
>  * @bdi: the device's backing_dev_info structure.
> @@ -275,8 +287,8 @@ static int move_expired_inodes(struct list_head *delaying_queue,
> 
> 	if ((flags & EXPIRE_DIRTY_ATIME) == 0)
> 		older_than_this = work->older_than_this;
> -	else if ((work->reason == WB_REASON_SYNC) == 0) {
> -		expire_time = jiffies - (HZ * 86400);
> +	else if (!work->for_sync) {
> +		expire_time = jiffies - (dirtytime_expire_interval * HZ);
> 		older_than_this = &expire_time;
> 	}
> 	while (!list_empty(delaying_queue)) {
> @@ -458,6 +470,7 @@ static void requeue_inode(struct inode *inode, struct bdi_writeback *wb,
> 		 */
> 		redirty_tail(inode, wb);
> 	} else if (inode->i_state & I_DIRTY_TIME) {
> +		inode->dirtied_when = jiffies;
> 		list_move(&inode->i_wb_list, &wb->b_dirty_time);
> 	} else {
> 		/* The inode is clean. Remove from writeback lists. */
> @@ -505,12 +518,17 @@ __writeback_single_inode(struct inode *inode, struct writeback_control *wbc)
> 	spin_lock(&inode->i_lock);
> 
> 	dirty = inode->i_state & I_DIRTY;
> -	if (((dirty & (I_DIRTY_SYNC | I_DIRTY_DATASYNC)) &&
> -	     (inode->i_state & I_DIRTY_TIME)) ||
> -	    (inode->i_state & I_DIRTY_TIME_EXPIRED)) {
> -		dirty |= I_DIRTY_TIME | I_DIRTY_TIME_EXPIRED;
> -		trace_writeback_lazytime(inode);
> -	}
> +	if (inode->i_state & I_DIRTY_TIME) {
> +		if ((dirty & (I_DIRTY_SYNC | I_DIRTY_DATASYNC)) ||
> +		    unlikely(inode->i_state & I_DIRTY_TIME_EXPIRED) ||
> +		    unlikely(time_after((inode->dirtied_time_when +
> +					 dirtytime_expire_interval * HZ),
> +					jiffies))) {
> +			dirty |= I_DIRTY_TIME | I_DIRTY_TIME_EXPIRED;
> +			trace_writeback_lazytime(inode);
> +		}
> +	} else
> +		inode->i_state &= ~I_DIRTY_TIME_EXPIRED;
> 	inode->i_state &= ~dirty;
> 
> 	/*
> @@ -1131,6 +1149,45 @@ void wakeup_flusher_threads(long nr_pages, enum wb_reason reason)
> 	rcu_read_unlock();
> }
> 
> +/*
> + * Wake up bdi's periodically to make sure dirtytime inodes gets
> + * written back periodically.  We deliberately do *not* check the
> + * b_dirtytime list in wb_has_dirty_io(), since this would cause the
> + * kernel to be constantly waking up once there are any dirtytime
> + * inodes on the system.  So instead we define a separate delayed work
> + * function which gets called much more rarely.  (By default, only
> + * once every 12 hours.)
> + *
> + * If there is any other write activity going on in the file system,
> + * this function won't be necessary.  But if the only thing that has
> + * happened on the file system is a dirtytime inode caused by an atime
> + * update, we need this infrastructure below to make sure that inode
> + * eventually gets pushed out to disk.
> + */
> +static void wakeup_dirtytime_writeback(struct work_struct *w);
> +static DECLARE_DELAYED_WORK(dirtytime_work, wakeup_dirtytime_writeback);
> +
> +static void wakeup_dirtytime_writeback(struct work_struct *w)
> +{
> +	struct backing_dev_info *bdi;
> +
> +	rcu_read_lock();
> +	list_for_each_entry_rcu(bdi, &bdi_list, bdi_list) {
> +		if (list_empty(&bdi->wb.b_dirty_time))
> +			continue;
> +		bdi_wakeup_thread(bdi);
> +	}
> +	rcu_read_unlock();
> +	schedule_delayed_work(&dirtytime_work, dirtytime_expire_interval * HZ);
> +}
> +
> +static int __init start_dirtytime_writeback(void)
> +{
> +	schedule_delayed_work(&dirtytime_work, dirtytime_expire_interval * HZ);
> +	return 0;
> +}
> +__initcall(start_dirtytime_writeback);
> +
> static noinline void block_dump___mark_inode_dirty(struct inode *inode)
> {
> 	if (inode->i_ino || strcmp(inode->i_sb->s_id, "bdev")) {
> @@ -1269,8 +1326,13 @@ void __mark_inode_dirty(struct inode *inode, int flags)
> 			}
> 
> 			inode->dirtied_when = jiffies;
> -			list_move(&inode->i_wb_list, dirtytime ?
> -				  &bdi->wb.b_dirty_time : &bdi->wb.b_dirty);
> +			if (dirtytime)
> +				inode->dirtied_time_when = jiffies;
> +			if (inode->i_state & (I_DIRTY_INODE | I_DIRTY_PAGES))
> +				list_move(&inode->i_wb_list, &bdi->wb.b_dirty);
> +			else
> +				list_move(&inode->i_wb_list,
> +					  &bdi->wb.b_dirty_time);
> 			spin_unlock(&bdi->wb.list_lock);
> 			trace_writeback_dirty_inode_enqueue(inode);
> 
> diff --git a/include/linux/fs.h b/include/linux/fs.h
> index b4d71b5..f4131e8 100644
> --- a/include/linux/fs.h
> +++ b/include/linux/fs.h
> @@ -604,6 +604,7 @@ struct inode {
> 	struct mutex		i_mutex;
> 
> 	unsigned long		dirtied_when;	/* jiffies of first dirtying */
> +	unsigned long		dirtied_time_when;

> 
> 	struct hlist_node	i_hash;
> 	struct list_head	i_wb_list;	/* backing dev IO list */
> -- 
> 2.3.0
> 
> --
> To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in
> the body of a message to majordomo@vger.kernel.org
> More majordomo info at  http://vger.kernel.org/majordomo-info.html


Cheers, Andreas





--
To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Jan Kara March 17, 2015, 10:29 a.m. UTC | #2
On Mon 16-03-15 15:14:19, Ted Tso wrote:
> Jan Kara pointed out that if there is an inode which is constantly
> getting dirtied with I_DIRTY_PAGES, an inode with an updated timestamp
> will never be written since inode->dirtied_when is constantly getting
> updated.  We fix this by adding an extra field to the inode,
> dirtied_time_when, so inodes with a stale dirtytime can get detected
> and handled.
> 
> In addition, if we have a dirtytime inode caused by an atime update,
> and there is no write activity on the file system, we need to have a
> secondary system to make sure these inodes get written out.  We do
> this by setting up a second delayed work structure which wakes up the
> CPU much more rarely compared to writeback_expire_centisecs.
> 
> Signed-off-by: Theodore Ts'o <tytso@mit.edu>
> Cc: stable@vger.kernel.org
  Since this was merged in rc1, I don't think CC to stable is needed.

> @@ -505,12 +518,17 @@ __writeback_single_inode(struct inode *inode, struct writeback_control *wbc)
>  	spin_lock(&inode->i_lock);
>  
>  	dirty = inode->i_state & I_DIRTY;
> -	if (((dirty & (I_DIRTY_SYNC | I_DIRTY_DATASYNC)) &&
> -	     (inode->i_state & I_DIRTY_TIME)) ||
> -	    (inode->i_state & I_DIRTY_TIME_EXPIRED)) {
> -		dirty |= I_DIRTY_TIME | I_DIRTY_TIME_EXPIRED;
> -		trace_writeback_lazytime(inode);
> -	}
> +	if (inode->i_state & I_DIRTY_TIME) {
> +		if ((dirty & (I_DIRTY_SYNC | I_DIRTY_DATASYNC)) ||
> +		    unlikely(inode->i_state & I_DIRTY_TIME_EXPIRED) ||
> +		    unlikely(time_after((inode->dirtied_time_when +
> +					 dirtytime_expire_interval * HZ),
> +					jiffies))) {
  The time comparison is the other way around, isn't it? After fixing that
feel free to add:
  Reviewed-by: Jan Kara <jack@suse.cz>

								Honza
Jan Kara March 17, 2015, 10:33 a.m. UTC | #3
On Mon 16-03-15 15:34:12, Andreas Dilger wrote:
> On Mar 16, 2015, at 1:14 PM, Theodore Ts'o <tytso@mit.edu> wrote:
> > 
> > Jan Kara pointed out that if there is an inode which is constantly
> > getting dirtied with I_DIRTY_PAGES, an inode with an updated timestamp
> > will never be written since inode->dirtied_when is constantly getting
> > updated.  We fix this by adding an extra field to the inode,
> > dirtied_time_when, so inodes with a stale dirtytime can get detected
> > and handled.
> 
> The drawback here is that this adds another 8 bytes to every inode for
> a field of marginal value, since this is only important for the rare
> case of a file that is being dirtied continuously.
  Yes.

> I wonder if something more lightweight could be added to avoid this
> problem?  For example, we only care about this case if it has been
> going on for more than the lazytime interval (about a day), so the
> inode could store a 16-bit i_dirtied_time_when that is approximately
> (jiffies >> bits_in_a_half_a_day) and only check time_after() that.
> The __u16 could fit into some existing hole (e.g. after i_bytes on my
> kernel) and avoid expanding the size of the inode at all.
> 
> The remaining high bits of i_dirtied_time_when would be irrelevant, since
> a __u16 of half-days is about 80 years, so it would be enough to compare:
> 
> 
>     time_after(i_dirtied_time_when, (__u16)(jiffies >> bits_in_half_a_day))
> 
> 
> A day is 86400s, so 43200s is close to (1 << 22) jiffies for HZ=100, and
> (1 << 25) jiffies is about 3/8 of a day for HZ=1000.  Since the exact
> times for inode writeout don't matter very much here, having only shifts
> to convert jiffies to i_dirtied_time_when in the kernel is better I think.
  Yes, something like this should be possible. But I wanted that to happen
as a separate patch once we have everything working correctly. The code is
subtle enough that I didn't want Ted to complicate it with further
optimizations initially.

> Minor issue, is there a good reason why dirtied_time_when doesn't have an
> "i_" prefix?
  I guess it's matching with dirtied_when which doesn't have i_ prefix just
because noone added it initially. I don't really care either way.

								Honza
Theodore Ts'o March 17, 2015, 3:09 p.m. UTC | #4
On Mon, Mar 16, 2015 at 03:34:12PM -0600, Andreas Dilger wrote:
> I wonder if something more lightweight could be added to avoid this
> problem?  For example, we only care about this case if it has been
> going on for more than the lazytime interval (about a day), so the
> inode could store a 16-bit i_dirtied_time_when that is approximately
> (jiffies >> bits_in_a_half_a_day) and only check time_after() that.
> The __u16 could fit into some existing hole (e.g. after i_bytes on my
> kernel) and avoid expanding the size of the inode at all.
> 
> The remaining high bits of i_dirtied_time_when would be irrelevant, since
> a __u16 of half-days is about 80 years, so it would be enough to compare:
> 
> 
>     time_after(i_dirtied_time_when, (__u16)(jiffies >> bits_in_half_a_day))

That won't work correctly; we'd have to do something like this

#define u16_after(a,b) (typecheck(__u16, a) && typecheck(__u16, b) && \
		 ((__s16)((b) - (a)) < 0))


> Minor issue, is there a good reason why dirtied_time_when doesn't have an
> "i_" prefix?

It's because dirtied_when also doesn't have an i_ prefix, but arguably
it should.

						- Ted
--
To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Theodore Ts'o March 17, 2015, 3:53 p.m. UTC | #5
On Tue, Mar 17, 2015 at 11:33:37AM +0100, Jan Kara wrote:
>   Yes, something like this should be possible. But I wanted that to happen
> as a separate patch once we have everything working correctly. The code is
> subtle enough that I didn't want Ted to complicate it with further
> optimizations initially.

Yes, agreed, I took a closer look at it and making the change that
Andreas suggested is more complicated than it first seems.

We can try to fix this later, but it's definitely messy.

							- Ted
--
To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
diff mbox

Patch

diff --git a/fs/fs-writeback.c b/fs/fs-writeback.c
index e907052..ae13fba 100644
--- a/fs/fs-writeback.c
+++ b/fs/fs-writeback.c
@@ -53,6 +53,18 @@  struct wb_writeback_work {
 	struct completion *done;	/* set if the caller waits */
 };
 
+/*
+ * If an inode is constantly having its pages dirtied, but then the
+ * updates stop dirtytime_expire_interval seconds in the past, it's
+ * possible for the worst case time between when an inode has its
+ * timestamps updated and when they finally get written out to be two
+ * dirtytime_expire_intervals.  We set the default to 12 hours (in
+ * seconds), which means most of the time inodes will have their
+ * timestamps written to disk after 12 hours, but in the worst case a
+ * few inodes might not their timestamps updated for 24 hours.
+ */
+unsigned int dirtytime_expire_interval = 12 * 60 * 60;
+
 /**
  * writeback_in_progress - determine whether there is writeback in progress
  * @bdi: the device's backing_dev_info structure.
@@ -275,8 +287,8 @@  static int move_expired_inodes(struct list_head *delaying_queue,
 
 	if ((flags & EXPIRE_DIRTY_ATIME) == 0)
 		older_than_this = work->older_than_this;
-	else if ((work->reason == WB_REASON_SYNC) == 0) {
-		expire_time = jiffies - (HZ * 86400);
+	else if (!work->for_sync) {
+		expire_time = jiffies - (dirtytime_expire_interval * HZ);
 		older_than_this = &expire_time;
 	}
 	while (!list_empty(delaying_queue)) {
@@ -458,6 +470,7 @@  static void requeue_inode(struct inode *inode, struct bdi_writeback *wb,
 		 */
 		redirty_tail(inode, wb);
 	} else if (inode->i_state & I_DIRTY_TIME) {
+		inode->dirtied_when = jiffies;
 		list_move(&inode->i_wb_list, &wb->b_dirty_time);
 	} else {
 		/* The inode is clean. Remove from writeback lists. */
@@ -505,12 +518,17 @@  __writeback_single_inode(struct inode *inode, struct writeback_control *wbc)
 	spin_lock(&inode->i_lock);
 
 	dirty = inode->i_state & I_DIRTY;
-	if (((dirty & (I_DIRTY_SYNC | I_DIRTY_DATASYNC)) &&
-	     (inode->i_state & I_DIRTY_TIME)) ||
-	    (inode->i_state & I_DIRTY_TIME_EXPIRED)) {
-		dirty |= I_DIRTY_TIME | I_DIRTY_TIME_EXPIRED;
-		trace_writeback_lazytime(inode);
-	}
+	if (inode->i_state & I_DIRTY_TIME) {
+		if ((dirty & (I_DIRTY_SYNC | I_DIRTY_DATASYNC)) ||
+		    unlikely(inode->i_state & I_DIRTY_TIME_EXPIRED) ||
+		    unlikely(time_after((inode->dirtied_time_when +
+					 dirtytime_expire_interval * HZ),
+					jiffies))) {
+			dirty |= I_DIRTY_TIME | I_DIRTY_TIME_EXPIRED;
+			trace_writeback_lazytime(inode);
+		}
+	} else
+		inode->i_state &= ~I_DIRTY_TIME_EXPIRED;
 	inode->i_state &= ~dirty;
 
 	/*
@@ -1131,6 +1149,45 @@  void wakeup_flusher_threads(long nr_pages, enum wb_reason reason)
 	rcu_read_unlock();
 }
 
+/*
+ * Wake up bdi's periodically to make sure dirtytime inodes gets
+ * written back periodically.  We deliberately do *not* check the
+ * b_dirtytime list in wb_has_dirty_io(), since this would cause the
+ * kernel to be constantly waking up once there are any dirtytime
+ * inodes on the system.  So instead we define a separate delayed work
+ * function which gets called much more rarely.  (By default, only
+ * once every 12 hours.)
+ *
+ * If there is any other write activity going on in the file system,
+ * this function won't be necessary.  But if the only thing that has
+ * happened on the file system is a dirtytime inode caused by an atime
+ * update, we need this infrastructure below to make sure that inode
+ * eventually gets pushed out to disk.
+ */
+static void wakeup_dirtytime_writeback(struct work_struct *w);
+static DECLARE_DELAYED_WORK(dirtytime_work, wakeup_dirtytime_writeback);
+
+static void wakeup_dirtytime_writeback(struct work_struct *w)
+{
+	struct backing_dev_info *bdi;
+
+	rcu_read_lock();
+	list_for_each_entry_rcu(bdi, &bdi_list, bdi_list) {
+		if (list_empty(&bdi->wb.b_dirty_time))
+			continue;
+		bdi_wakeup_thread(bdi);
+	}
+	rcu_read_unlock();
+	schedule_delayed_work(&dirtytime_work, dirtytime_expire_interval * HZ);
+}
+
+static int __init start_dirtytime_writeback(void)
+{
+	schedule_delayed_work(&dirtytime_work, dirtytime_expire_interval * HZ);
+	return 0;
+}
+__initcall(start_dirtytime_writeback);
+
 static noinline void block_dump___mark_inode_dirty(struct inode *inode)
 {
 	if (inode->i_ino || strcmp(inode->i_sb->s_id, "bdev")) {
@@ -1269,8 +1326,13 @@  void __mark_inode_dirty(struct inode *inode, int flags)
 			}
 
 			inode->dirtied_when = jiffies;
-			list_move(&inode->i_wb_list, dirtytime ?
-				  &bdi->wb.b_dirty_time : &bdi->wb.b_dirty);
+			if (dirtytime)
+				inode->dirtied_time_when = jiffies;
+			if (inode->i_state & (I_DIRTY_INODE | I_DIRTY_PAGES))
+				list_move(&inode->i_wb_list, &bdi->wb.b_dirty);
+			else
+				list_move(&inode->i_wb_list,
+					  &bdi->wb.b_dirty_time);
 			spin_unlock(&bdi->wb.list_lock);
 			trace_writeback_dirty_inode_enqueue(inode);
 
diff --git a/include/linux/fs.h b/include/linux/fs.h
index b4d71b5..f4131e8 100644
--- a/include/linux/fs.h
+++ b/include/linux/fs.h
@@ -604,6 +604,7 @@  struct inode {
 	struct mutex		i_mutex;
 
 	unsigned long		dirtied_when;	/* jiffies of first dirtying */
+	unsigned long		dirtied_time_when;
 
 	struct hlist_node	i_hash;
 	struct list_head	i_wb_list;	/* backing dev IO list */