diff options
Diffstat (limited to 'drivers/md/raid5.c')
| -rw-r--r-- | drivers/md/raid5.c | 21 | 
1 files changed, 21 insertions, 0 deletions
| diff --git a/drivers/md/raid5.c b/drivers/md/raid5.c index f535ce2c267a..90638ba67812 100644 --- a/drivers/md/raid5.c +++ b/drivers/md/raid5.c @@ -228,6 +228,16 @@ static void do_release_stripe(struct r5conf *conf, struct stripe_head *sh,  		for (i = sh->disks; i--; )  			if (test_bit(R5_InJournal, &sh->dev[i].flags))  				injournal++; +	/* +	 * When quiesce in r5c write back, set STRIPE_HANDLE for stripes with +	 * data in journal, so they are not released to cached lists +	 */ +	if (conf->quiesce && r5c_is_writeback(conf->log) && +	    !test_bit(STRIPE_HANDLE, &sh->state) && injournal != 0) { +		if (test_bit(STRIPE_R5C_CACHING, &sh->state)) +			r5c_make_stripe_write_out(sh); +		set_bit(STRIPE_HANDLE, &sh->state); +	}  	if (test_bit(STRIPE_HANDLE, &sh->state)) {  		if (test_bit(STRIPE_DELAYED, &sh->state) && @@ -268,6 +278,7 @@ static void do_release_stripe(struct r5conf *conf, struct stripe_head *sh,  					if (test_and_clear_bit(STRIPE_R5C_PARTIAL_STRIPE, &sh->state))  						atomic_dec(&conf->r5c_cached_partial_stripes);  					list_add_tail(&sh->lru, &conf->r5c_full_stripe_list); +					r5c_check_cached_full_stripe(conf);  				} else {  					/* partial stripe */  					if (!test_and_set_bit(STRIPE_R5C_PARTIAL_STRIPE, @@ -639,9 +650,12 @@ raid5_get_active_stripe(struct r5conf *conf, sector_t sector,  			}  			if (noblock && sh == NULL)  				break; + +			r5c_check_stripe_cache_usage(conf);  			if (!sh) {  				set_bit(R5_INACTIVE_BLOCKED,  					&conf->cache_state); +				r5l_wake_reclaim(conf->log, 0);  				wait_event_lock_irq(  					conf->wait_for_stripe,  					!list_empty(conf->inactive_list + hash) && @@ -1992,7 +2006,9 @@ static struct stripe_head *alloc_stripe(struct kmem_cache *sc, gfp_t gfp,  		spin_lock_init(&sh->batch_lock);  		INIT_LIST_HEAD(&sh->batch_list);  		INIT_LIST_HEAD(&sh->lru); +		INIT_LIST_HEAD(&sh->r5c);  		atomic_set(&sh->count, 1); +		sh->log_start = MaxSector;  		for (i = 0; i < disks; i++) {  			struct r5dev *dev = &sh->dev[i]; @@ -4759,6 +4775,10 @@ static int raid5_congested(struct mddev *mddev, int bits)  	if (test_bit(R5_INACTIVE_BLOCKED, &conf->cache_state))  		return 1; + +	/* Also checks whether there is pressure on r5cache log space */ +	if (test_bit(R5C_LOG_TIGHT, &conf->cache_state)) +		return 1;  	if (conf->quiesce)  		return 1;  	if (atomic_read(&conf->empty_inactive_list_nr)) @@ -7661,6 +7681,7 @@ static void raid5_quiesce(struct mddev *mddev, int state)  		/* '2' tells resync/reshape to pause so that all  		 * active stripes can drain  		 */ +		r5c_flush_cache(conf, INT_MAX);  		conf->quiesce = 2;  		wait_event_cmd(conf->wait_for_quiescent,  				    atomic_read(&conf->active_stripes) == 0 && | 
