1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-03 05:18:29 +03:00

Mirror: Fix issue preventing PV creation on mirror LVs

Commit b248ba0a39 attempted to
prevent mirror devices which had a failed device in their
mirrored log from being usable/readable by LVM.  This was to
protect against circular dependancies where one LVM command
could be blocked trying to read one of these affected mirrors
while the LVM command to fix/unblock that mirror was stuck
behind the currently running command.

The above commit went wrong when it used 'device_is_usable()' to
recurse on the mirrored log device to check if it was suspended
or blocked.  The 'device_is_usable' function also contains a check
for reserved names - like *_mlog, etc.  This last check always
triggered when checking a mirror's log simply because of the name,
not because it was suspended or blocked - a false positive.

The solution is to create a new function like 'device_is_usable',
but without the check for reserved names.  Using this new function
(device_is_suspended_or_blocked), we can check the status of a
mirror's log device properly.
This commit is contained in:
Jonathan Brassow 2013-08-07 17:42:26 -05:00
parent b15278c3dc
commit c95f17ea64
2 changed files with 40 additions and 4 deletions

View File

@ -167,10 +167,18 @@ int pv_uses_vg(struct physical_volume *pv,
struct volume_group *vg); struct volume_group *vg);
/* /*
* Returns 1 if mapped device is not suspended. * Returns 1 if mapped device is not suspended, blocked or
* is using a reserved name.
*/ */
int device_is_usable(struct device *dev); int device_is_usable(struct device *dev);
/*
* Returns 1 if the device is suspended or blocking.
* (Does not perform check on the LV name of the device.)
* N.B. This is !device_is_usable() without the name check.
*/
int device_is_suspended_or_blocking(struct device *dev);
/* /*
* Declaration moved here from fs.h to keep header fs.h hidden * Declaration moved here from fs.h to keep header fs.h hidden
*/ */

View File

@ -275,7 +275,7 @@ static int _ignore_blocked_mirror_devices(struct device *dev,
goto_out; goto_out;
tmp_dev->dev = log_dev; tmp_dev->dev = log_dev;
if (!device_is_usable(tmp_dev)) if (device_is_suspended_or_blocking(tmp_dev))
goto_out; goto_out;
} }
} }
@ -329,7 +329,24 @@ out:
return r; return r;
} }
int device_is_usable(struct device *dev) /*
* _device_is_usable
* @dev
* @check_lv_names
*
* A device is considered not usable if it is:
* 1) An empty device (no targets)
* 2) A blocked mirror (i.e. a mirror with a failure and block_on_error set)
* 3) ignore_suspended_devices is set and
* a) the device is suspended
* b) it is a snapshot origin
* 4) an error target
* And optionally, if 'check_lv_names' is set
* 5) the LV name is a reserved name.
*
* Returns: 1 if usable, 0 otherwise
*/
static int _device_is_usable(struct device *dev, int check_lv_names)
{ {
struct dm_task *dmt; struct dm_task *dmt;
struct dm_info info; struct dm_info info;
@ -416,7 +433,8 @@ int device_is_usable(struct device *dev)
/* FIXME Also check dependencies? */ /* FIXME Also check dependencies? */
/* Check internal lvm devices */ /* Check internal lvm devices */
if (uuid && !strncmp(uuid, UUID_PREFIX, sizeof(UUID_PREFIX) - 1)) { if (check_lv_names &&
uuid && !strncmp(uuid, UUID_PREFIX, sizeof(UUID_PREFIX) - 1)) {
if (!(vgname = dm_strdup(name)) || if (!(vgname = dm_strdup(name)) ||
!dm_split_lvm_name(NULL, NULL, &vgname, &lvname, &layer)) !dm_split_lvm_name(NULL, NULL, &vgname, &lvname, &layer))
goto_out; goto_out;
@ -436,6 +454,16 @@ int device_is_usable(struct device *dev)
return r; return r;
} }
int device_is_usable(struct device *dev)
{
return _device_is_usable(dev, 1);
}
int device_is_suspended_or_blocking(struct device *dev)
{
return !_device_is_usable(dev, 0);
}
static int _info(const char *dlid, int with_open_count, int with_read_ahead, static int _info(const char *dlid, int with_open_count, int with_read_ahead,
struct dm_info *info, uint32_t *read_ahead) struct dm_info *info, uint32_t *read_ahead)
{ {