Commit da7d9676 authored by NeilBrown's avatar NeilBrown Committed by Greg Kroah-Hartman

md: manage redundancy group in sysfs when changing level.

commit a64c876f upstream.

Some levels expect the 'redundancy group' to be present,
others don't.
So when we change level of an array we might need to
add or remove this group.

This requires fixing up the current practice of overloading ->private
to indicate (when ->pers == NULL) that something needs to be removed.
So create a new ->to_remove to fill that role.

When changing levels, we may need to add or remove attributes.  When
changing RAID5 -> RAID6, we both add and remove the same thing.  It is
important to catch this and optimise it out as the removal is delayed
until a lock is released, so trying to add immediately would cause
problems.
Signed-off-by: default avatarNeilBrown <neilb@suse.de>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent 3bbcbb8c
...@@ -510,9 +510,9 @@ static inline int mddev_trylock(mddev_t * mddev) ...@@ -510,9 +510,9 @@ static inline int mddev_trylock(mddev_t * mddev)
static struct attribute_group md_redundancy_group; static struct attribute_group md_redundancy_group;
static inline void mddev_unlock(mddev_t * mddev) static void mddev_unlock(mddev_t * mddev)
{ {
if (mddev->pers == NULL && mddev->private) { if (mddev->to_remove) {
/* These cannot be removed under reconfig_mutex as /* These cannot be removed under reconfig_mutex as
* an access to the files will try to take reconfig_mutex * an access to the files will try to take reconfig_mutex
* while holding the file unremovable, which leads to * while holding the file unremovable, which leads to
...@@ -521,16 +521,20 @@ static inline void mddev_unlock(mddev_t * mddev) ...@@ -521,16 +521,20 @@ static inline void mddev_unlock(mddev_t * mddev)
* it while holding reconfig_mutex, and md_run can * it while holding reconfig_mutex, and md_run can
* use it to wait for the remove to complete. * use it to wait for the remove to complete.
*/ */
struct attribute_group *to_remove = mddev->to_remove;
mddev->to_remove = NULL;
mutex_lock(&mddev->open_mutex); mutex_lock(&mddev->open_mutex);
mutex_unlock(&mddev->reconfig_mutex); mutex_unlock(&mddev->reconfig_mutex);
if (to_remove != &md_redundancy_group)
sysfs_remove_group(&mddev->kobj, to_remove);
if (mddev->pers == NULL ||
mddev->pers->sync_request == NULL) {
sysfs_remove_group(&mddev->kobj, &md_redundancy_group); sysfs_remove_group(&mddev->kobj, &md_redundancy_group);
if (mddev->private != (void*)1)
sysfs_remove_group(&mddev->kobj, mddev->private);
if (mddev->sysfs_action) if (mddev->sysfs_action)
sysfs_put(mddev->sysfs_action); sysfs_put(mddev->sysfs_action);
mddev->sysfs_action = NULL; mddev->sysfs_action = NULL;
mddev->private = NULL; }
mutex_unlock(&mddev->open_mutex); mutex_unlock(&mddev->open_mutex);
} else } else
mutex_unlock(&mddev->reconfig_mutex); mutex_unlock(&mddev->reconfig_mutex);
...@@ -3003,6 +3007,23 @@ level_store(mddev_t *mddev, const char *buf, size_t len) ...@@ -3003,6 +3007,23 @@ level_store(mddev_t *mddev, const char *buf, size_t len)
/* Looks like we have a winner */ /* Looks like we have a winner */
mddev_suspend(mddev); mddev_suspend(mddev);
mddev->pers->stop(mddev); mddev->pers->stop(mddev);
if (mddev->pers->sync_request == NULL &&
pers->sync_request != NULL) {
/* need to add the md_redundancy_group */
if (sysfs_create_group(&mddev->kobj, &md_redundancy_group))
printk(KERN_WARNING
"md: cannot register extra attributes for %s\n",
mdname(mddev));
mddev->sysfs_action = sysfs_get_dirent(mddev->kobj.sd, "sync_action");
}
if (mddev->pers->sync_request != NULL &&
pers->sync_request == NULL) {
/* need to remove the md_redundancy_group */
if (mddev->to_remove == NULL)
mddev->to_remove = &md_redundancy_group;
}
module_put(mddev->pers->owner); module_put(mddev->pers->owner);
/* Invalidate devices that are now superfluous */ /* Invalidate devices that are now superfluous */
list_for_each_entry(rdev, &mddev->disks, same_set) list_for_each_entry(rdev, &mddev->disks, same_set)
...@@ -4557,8 +4578,8 @@ static int do_md_stop(mddev_t * mddev, int mode, int is_open) ...@@ -4557,8 +4578,8 @@ static int do_md_stop(mddev_t * mddev, int mode, int is_open)
mddev->queue->unplug_fn = NULL; mddev->queue->unplug_fn = NULL;
mddev->queue->backing_dev_info.congested_fn = NULL; mddev->queue->backing_dev_info.congested_fn = NULL;
module_put(mddev->pers->owner); module_put(mddev->pers->owner);
if (mddev->pers->sync_request && mddev->private == NULL) if (mddev->pers->sync_request && mddev->to_remove == NULL)
mddev->private = (void*)1; mddev->to_remove = &md_redundancy_group;
mddev->pers = NULL; mddev->pers = NULL;
/* tell userspace to handle 'inactive' */ /* tell userspace to handle 'inactive' */
sysfs_notify_dirent(mddev->sysfs_state); sysfs_notify_dirent(mddev->sysfs_state);
......
...@@ -305,6 +305,7 @@ struct mddev_s ...@@ -305,6 +305,7 @@ struct mddev_s
atomic_t max_corr_read_errors; /* max read retries */ atomic_t max_corr_read_errors; /* max read retries */
struct list_head all_mddevs; struct list_head all_mddevs;
struct attribute_group *to_remove;
/* Generic barrier handling. /* Generic barrier handling.
* If there is a pending barrier request, all other * If there is a pending barrier request, all other
* writes are blocked while the devices are flushed. * writes are blocked while the devices are flushed.
......
...@@ -5087,7 +5087,9 @@ static int run(mddev_t *mddev) ...@@ -5087,7 +5087,9 @@ static int run(mddev_t *mddev)
} }
/* Ok, everything is just fine now */ /* Ok, everything is just fine now */
if (sysfs_create_group(&mddev->kobj, &raid5_attrs_group)) if (mddev->to_remove == &raid5_attrs_group)
mddev->to_remove = NULL;
else if (sysfs_create_group(&mddev->kobj, &raid5_attrs_group))
printk(KERN_WARNING printk(KERN_WARNING
"raid5: failed to create sysfs attributes for %s\n", "raid5: failed to create sysfs attributes for %s\n",
mdname(mddev)); mdname(mddev));
...@@ -5134,7 +5136,8 @@ static int stop(mddev_t *mddev) ...@@ -5134,7 +5136,8 @@ static int stop(mddev_t *mddev)
mddev->queue->backing_dev_info.congested_fn = NULL; mddev->queue->backing_dev_info.congested_fn = NULL;
blk_sync_queue(mddev->queue); /* the unplug fn references 'conf'*/ blk_sync_queue(mddev->queue); /* the unplug fn references 'conf'*/
free_conf(conf); free_conf(conf);
mddev->private = &raid5_attrs_group; mddev->private = NULL;
mddev->to_remove = &raid5_attrs_group;
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment