md: remove special meaning of ->quiesce(.., 2)

The '2' argument means "wake up anything that is waiting".
This is an inelegant part of the design and was added
to help support management of suspend_lo/suspend_hi setting.
Now that suspend_lo/hi is managed in mddev_suspend/resume,
that need is gone.
These is still a couple of places where we call 'quiesce'
with an argument of '2', but they can safely be changed to
call ->quiesce(.., 1); ->quiesce(.., 0) which
achieve the same result at the small cost of pausing IO
briefly.

This removes a small "optimization" from suspend_{hi,lo}_store,
but it isn't clear that optimization served a useful purpose.
The code now is a lot clearer.

Suggested-by: Shaohua Li <shli@kernel.org>
Signed-off-by: NeilBrown <neilb@suse.com>
Signed-off-by: Shaohua Li <shli@fb.com>
This commit is contained in:
NeilBrown 2017-10-19 12:49:15 +11:00 committed by Shaohua Li
parent 35bfc52187
commit b03e0ccb5a
9 changed files with 37 additions and 69 deletions

View File

@ -442,10 +442,11 @@ static void __remove_suspend_info(struct md_cluster_info *cinfo, int slot)
static void remove_suspend_info(struct mddev *mddev, int slot) static void remove_suspend_info(struct mddev *mddev, int slot)
{ {
struct md_cluster_info *cinfo = mddev->cluster_info; struct md_cluster_info *cinfo = mddev->cluster_info;
mddev->pers->quiesce(mddev, 1);
spin_lock_irq(&cinfo->suspend_lock); spin_lock_irq(&cinfo->suspend_lock);
__remove_suspend_info(cinfo, slot); __remove_suspend_info(cinfo, slot);
spin_unlock_irq(&cinfo->suspend_lock); spin_unlock_irq(&cinfo->suspend_lock);
mddev->pers->quiesce(mddev, 2); mddev->pers->quiesce(mddev, 0);
} }
@ -492,13 +493,12 @@ static void process_suspend_info(struct mddev *mddev,
s->lo = lo; s->lo = lo;
s->hi = hi; s->hi = hi;
mddev->pers->quiesce(mddev, 1); mddev->pers->quiesce(mddev, 1);
mddev->pers->quiesce(mddev, 0);
spin_lock_irq(&cinfo->suspend_lock); spin_lock_irq(&cinfo->suspend_lock);
/* Remove existing entry (if exists) before adding */ /* Remove existing entry (if exists) before adding */
__remove_suspend_info(cinfo, slot); __remove_suspend_info(cinfo, slot);
list_add(&s->list, &cinfo->suspend_list); list_add(&s->list, &cinfo->suspend_list);
spin_unlock_irq(&cinfo->suspend_lock); spin_unlock_irq(&cinfo->suspend_lock);
mddev->pers->quiesce(mddev, 2); mddev->pers->quiesce(mddev, 0);
} }
static void process_add_new_disk(struct mddev *mddev, struct cluster_msg *cmsg) static void process_add_new_disk(struct mddev *mddev, struct cluster_msg *cmsg)

View File

@ -4846,7 +4846,7 @@ suspend_lo_show(struct mddev *mddev, char *page)
static ssize_t static ssize_t
suspend_lo_store(struct mddev *mddev, const char *buf, size_t len) suspend_lo_store(struct mddev *mddev, const char *buf, size_t len)
{ {
unsigned long long old, new; unsigned long long new;
int err; int err;
err = kstrtoull(buf, 10, &new); err = kstrtoull(buf, 10, &new);
@ -4862,17 +4862,10 @@ suspend_lo_store(struct mddev *mddev, const char *buf, size_t len)
if (mddev->pers == NULL || if (mddev->pers == NULL ||
mddev->pers->quiesce == NULL) mddev->pers->quiesce == NULL)
goto unlock; goto unlock;
old = mddev->suspend_lo; mddev_suspend(mddev);
mddev->suspend_lo = new; mddev->suspend_lo = new;
if (new >= old) { mddev_resume(mddev);
/* Shrinking suspended region */
wake_up(&mddev->sb_wait);
mddev->pers->quiesce(mddev, 2);
} else {
/* Expanding suspended region - need to wait */
mddev_suspend(mddev);
mddev_resume(mddev);
}
err = 0; err = 0;
unlock: unlock:
mddev_unlock(mddev); mddev_unlock(mddev);
@ -4890,7 +4883,7 @@ suspend_hi_show(struct mddev *mddev, char *page)
static ssize_t static ssize_t
suspend_hi_store(struct mddev *mddev, const char *buf, size_t len) suspend_hi_store(struct mddev *mddev, const char *buf, size_t len)
{ {
unsigned long long old, new; unsigned long long new;
int err; int err;
err = kstrtoull(buf, 10, &new); err = kstrtoull(buf, 10, &new);
@ -4903,20 +4896,13 @@ suspend_hi_store(struct mddev *mddev, const char *buf, size_t len)
if (err) if (err)
return err; return err;
err = -EINVAL; err = -EINVAL;
if (mddev->pers == NULL || if (mddev->pers == NULL)
mddev->pers->quiesce == NULL)
goto unlock; goto unlock;
old = mddev->suspend_hi;
mddev_suspend(mddev);
mddev->suspend_hi = new; mddev->suspend_hi = new;
if (new <= old) { mddev_resume(mddev);
/* Shrinking suspended region */
wake_up(&mddev->sb_wait);
mddev->pers->quiesce(mddev, 2);
} else {
/* Expanding suspended region - need to wait */
mddev_suspend(mddev);
mddev_resume(mddev);
}
err = 0; err = 0;
unlock: unlock:
mddev_unlock(mddev); mddev_unlock(mddev);

View File

@ -544,12 +544,11 @@ struct md_personality
int (*check_reshape) (struct mddev *mddev); int (*check_reshape) (struct mddev *mddev);
int (*start_reshape) (struct mddev *mddev); int (*start_reshape) (struct mddev *mddev);
void (*finish_reshape) (struct mddev *mddev); void (*finish_reshape) (struct mddev *mddev);
/* quiesce moves between quiescence states /* quiesce suspends or resumes internal processing.
* 0 - fully active * 1 - stop new actions and wait for action io to complete
* 1 - no new requests allowed * 0 - return to normal behaviour
* others - reserved
*/ */
void (*quiesce) (struct mddev *mddev, int state); void (*quiesce) (struct mddev *mddev, int quiesce);
/* takeover is used to transition an array from one /* takeover is used to transition an array from one
* personality to another. The new personality must be able * personality to another. The new personality must be able
* to handle the data in the current layout. * to handle the data in the current layout.

View File

@ -768,7 +768,7 @@ static void *raid0_takeover(struct mddev *mddev)
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
} }
static void raid0_quiesce(struct mddev *mddev, int state) static void raid0_quiesce(struct mddev *mddev, int quiesce)
{ {
} }

View File

@ -3273,21 +3273,14 @@ static int raid1_reshape(struct mddev *mddev)
return 0; return 0;
} }
static void raid1_quiesce(struct mddev *mddev, int state) static void raid1_quiesce(struct mddev *mddev, int quiesce)
{ {
struct r1conf *conf = mddev->private; struct r1conf *conf = mddev->private;
switch(state) { if (quiesce)
case 2: /* wake for suspend */
wake_up(&conf->wait_barrier);
break;
case 1:
freeze_array(conf, 0); freeze_array(conf, 0);
break; else
case 0:
unfreeze_array(conf); unfreeze_array(conf);
break;
}
} }
static void *raid1_takeover(struct mddev *mddev) static void *raid1_takeover(struct mddev *mddev)

View File

@ -3828,18 +3828,14 @@ static void raid10_free(struct mddev *mddev, void *priv)
kfree(conf); kfree(conf);
} }
static void raid10_quiesce(struct mddev *mddev, int state) static void raid10_quiesce(struct mddev *mddev, int quiesce)
{ {
struct r10conf *conf = mddev->private; struct r10conf *conf = mddev->private;
switch(state) { if (quiesce)
case 1:
raise_barrier(conf, 0); raise_barrier(conf, 0);
break; else
case 0:
lower_barrier(conf); lower_barrier(conf);
break;
}
} }
static int raid10_resize(struct mddev *mddev, sector_t sectors) static int raid10_resize(struct mddev *mddev, sector_t sectors)

View File

@ -1589,21 +1589,21 @@ void r5l_wake_reclaim(struct r5l_log *log, sector_t space)
md_wakeup_thread(log->reclaim_thread); md_wakeup_thread(log->reclaim_thread);
} }
void r5l_quiesce(struct r5l_log *log, int state) void r5l_quiesce(struct r5l_log *log, int quiesce)
{ {
struct mddev *mddev; struct mddev *mddev;
if (!log || state == 2) if (!log)
return; return;
if (state == 0)
kthread_unpark(log->reclaim_thread->tsk); if (quiesce) {
else if (state == 1) {
/* make sure r5l_write_super_and_discard_space exits */ /* make sure r5l_write_super_and_discard_space exits */
mddev = log->rdev->mddev; mddev = log->rdev->mddev;
wake_up(&mddev->sb_wait); wake_up(&mddev->sb_wait);
kthread_park(log->reclaim_thread->tsk); kthread_park(log->reclaim_thread->tsk);
r5l_wake_reclaim(log, MaxSector); r5l_wake_reclaim(log, MaxSector);
r5l_do_reclaim(log); r5l_do_reclaim(log);
} } else
kthread_unpark(log->reclaim_thread->tsk);
} }
bool r5l_log_disk_error(struct r5conf *conf) bool r5l_log_disk_error(struct r5conf *conf)

View File

@ -8,7 +8,7 @@ extern void r5l_write_stripe_run(struct r5l_log *log);
extern void r5l_flush_stripe_to_raid(struct r5l_log *log); extern void r5l_flush_stripe_to_raid(struct r5l_log *log);
extern void r5l_stripe_write_finished(struct stripe_head *sh); extern void r5l_stripe_write_finished(struct stripe_head *sh);
extern int r5l_handle_flush_request(struct r5l_log *log, struct bio *bio); extern int r5l_handle_flush_request(struct r5l_log *log, struct bio *bio);
extern void r5l_quiesce(struct r5l_log *log, int state); extern void r5l_quiesce(struct r5l_log *log, int quiesce);
extern bool r5l_log_disk_error(struct r5conf *conf); extern bool r5l_log_disk_error(struct r5conf *conf);
extern bool r5c_is_writeback(struct r5l_log *log); extern bool r5c_is_writeback(struct r5l_log *log);
extern int extern int

View File

@ -8008,16 +8008,12 @@ static void raid5_finish_reshape(struct mddev *mddev)
} }
} }
static void raid5_quiesce(struct mddev *mddev, int state) static void raid5_quiesce(struct mddev *mddev, int quiesce)
{ {
struct r5conf *conf = mddev->private; struct r5conf *conf = mddev->private;
switch(state) { if (quiesce) {
case 2: /* resume for a suspend */ /* stop all writes */
wake_up(&conf->wait_for_overlap);
break;
case 1: /* stop all writes */
lock_all_device_hash_locks_irq(conf); lock_all_device_hash_locks_irq(conf);
/* '2' tells resync/reshape to pause so that all /* '2' tells resync/reshape to pause so that all
* active stripes can drain * active stripes can drain
@ -8033,17 +8029,15 @@ static void raid5_quiesce(struct mddev *mddev, int state)
unlock_all_device_hash_locks_irq(conf); unlock_all_device_hash_locks_irq(conf);
/* allow reshape to continue */ /* allow reshape to continue */
wake_up(&conf->wait_for_overlap); wake_up(&conf->wait_for_overlap);
break; } else {
/* re-enable writes */
case 0: /* re-enable writes */
lock_all_device_hash_locks_irq(conf); lock_all_device_hash_locks_irq(conf);
conf->quiesce = 0; conf->quiesce = 0;
wake_up(&conf->wait_for_quiescent); wake_up(&conf->wait_for_quiescent);
wake_up(&conf->wait_for_overlap); wake_up(&conf->wait_for_overlap);
unlock_all_device_hash_locks_irq(conf); unlock_all_device_hash_locks_irq(conf);
break;
} }
r5l_quiesce(conf->log, state); r5l_quiesce(conf->log, quiesce);
} }
static void *raid45_takeover_raid0(struct mddev *mddev, int level) static void *raid45_takeover_raid0(struct mddev *mddev, int level)