diff options
Diffstat (limited to 'fs/xfs/xfs_icache.c')
| -rw-r--r-- | fs/xfs/xfs_icache.c | 40 | 
1 files changed, 33 insertions, 7 deletions
| diff --git a/fs/xfs/xfs_icache.c b/fs/xfs/xfs_icache.c index 351849fc18ff..0f60e301eb1f 100644 --- a/fs/xfs/xfs_icache.c +++ b/fs/xfs/xfs_icache.c @@ -435,18 +435,23 @@ xfs_iget_check_free_state(  }  /* Make all pending inactivation work start immediately. */ -static void +static bool  xfs_inodegc_queue_all(  	struct xfs_mount	*mp)  {  	struct xfs_inodegc	*gc;  	int			cpu; +	bool			ret = false;  	for_each_online_cpu(cpu) {  		gc = per_cpu_ptr(mp->m_inodegc, cpu); -		if (!llist_empty(&gc->list)) +		if (!llist_empty(&gc->list)) {  			mod_delayed_work_on(cpu, mp->m_inodegc_wq, &gc->work, 0); +			ret = true; +		}  	} + +	return ret;  }  /* @@ -1856,6 +1861,8 @@ xfs_inodegc_worker(  	struct xfs_inode	*ip, *n;  	unsigned int		nofs_flag; +	ASSERT(gc->cpu == smp_processor_id()); +  	WRITE_ONCE(gc->items, 0);  	if (!node) @@ -1909,24 +1916,41 @@ xfs_inodegc_flush(  /*   * Flush all the pending work and then disable the inode inactivation background - * workers and wait for them to stop. + * workers and wait for them to stop.  Caller must hold sb->s_umount to + * coordinate changes in the inodegc_enabled state.   */  void  xfs_inodegc_stop(  	struct xfs_mount	*mp)  { +	bool			rerun; +  	if (!xfs_clear_inodegc_enabled(mp))  		return; +	/* +	 * Drain all pending inodegc work, including inodes that could be +	 * queued by racing xfs_inodegc_queue or xfs_inodegc_shrinker_scan +	 * threads that sample the inodegc state just prior to us clearing it. +	 * The inodegc flag state prevents new threads from queuing more +	 * inodes, so we queue pending work items and flush the workqueue until +	 * all inodegc lists are empty.  IOWs, we cannot use drain_workqueue +	 * here because it does not allow other unserialized mechanisms to +	 * reschedule inodegc work while this draining is in progress. +	 */  	xfs_inodegc_queue_all(mp); -	drain_workqueue(mp->m_inodegc_wq); +	do { +		flush_workqueue(mp->m_inodegc_wq); +		rerun = xfs_inodegc_queue_all(mp); +	} while (rerun);  	trace_xfs_inodegc_stop(mp, __return_address);  }  /*   * Enable the inode inactivation background workers and schedule deferred inode - * inactivation work if there is any. + * inactivation work if there is any.  Caller must hold sb->s_umount to + * coordinate changes in the inodegc_enabled state.   */  void  xfs_inodegc_start( @@ -2069,7 +2093,8 @@ xfs_inodegc_queue(  		queue_delay = 0;  	trace_xfs_inodegc_queue(mp, __return_address); -	mod_delayed_work(mp->m_inodegc_wq, &gc->work, queue_delay); +	mod_delayed_work_on(current_cpu(), mp->m_inodegc_wq, &gc->work, +			queue_delay);  	put_cpu_ptr(gc);  	if (xfs_inodegc_want_flush_work(ip, items, shrinker_hits)) { @@ -2113,7 +2138,8 @@ xfs_inodegc_cpu_dead(  	if (xfs_is_inodegc_enabled(mp)) {  		trace_xfs_inodegc_queue(mp, __return_address); -		mod_delayed_work(mp->m_inodegc_wq, &gc->work, 0); +		mod_delayed_work_on(current_cpu(), mp->m_inodegc_wq, &gc->work, +				0);  	}  	put_cpu_ptr(gc);  } | 
