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:
parent
35bfc52187
commit
b03e0ccb5a
|
@ -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)
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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.
|
||||||
|
|
|
@ -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)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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)
|
||||||
|
|
Loading…
Reference in New Issue