summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAlasdair G Kergon <agk@redhat.com>2017-05-23 01:37:14 +0100
committerAlasdair G Kergon <agk@redhat.com>2017-05-23 01:38:02 +0100
commitfbe7464df5018d8ab85dd312fae8c13796eca59d (patch)
tree6a957ef9c37cb89889f0b6be320a7ec88c09dd59
parentd1ddfc408535b9c4df432273657f952c59f16232 (diff)
downloadlvm2-fbe7464df5018d8ab85dd312fae8c13796eca59d.tar.gz
metadata: Unlock VG on more _vg_make_handle error paths
Internal error: VG lock vg0 must be requested before vg3, not after. Internal error: 3 device(s) were left open and have been closed.
-rw-r--r--lib/metadata/metadata.c26
1 files changed, 19 insertions, 7 deletions
diff --git a/lib/metadata/metadata.c b/lib/metadata/metadata.c
index a95b9e3da..36bb0e485 100644
--- a/lib/metadata/metadata.c
+++ b/lib/metadata/metadata.c
@@ -1089,8 +1089,7 @@ static struct volume_group *_vg_make_handle(struct cmd_context *cmd,
if (!vg && !(vg = alloc_vg("vg_make_handle", cmd, NULL)))
return_NULL;
- if (vg->read_status != failure)
- vg->read_status = failure;
+ vg->read_status = failure;
if (vg->fid && !_vg_update_vg_committed(vg))
vg->read_status |= FAILED_ALLOCATION;
@@ -1122,6 +1121,7 @@ int vg_has_unknown_segments(const struct volume_group *vg)
struct volume_group *vg_lock_and_create(struct cmd_context *cmd, const char *vg_name)
{
uint32_t rc;
+ struct volume_group *vg;
if (!validate_name(vg_name)) {
log_error("Invalid vg name %s", vg_name);
@@ -1134,7 +1134,11 @@ struct volume_group *vg_lock_and_create(struct cmd_context *cmd, const char *vg_
/* NOTE: let caller decide - this may be check for existence */
return _vg_make_handle(cmd, NULL, rc);
- return vg_create(cmd, vg_name);
+ vg = vg_create(cmd, vg_name);
+ if (!vg || vg_read_error(vg))
+ unlock_vg(cmd, NULL, vg_name);
+
+ return vg;
}
/*
@@ -5877,7 +5881,11 @@ static struct volume_group *_vg_lock_and_read(struct cmd_context *cmd, const cha
if (failure)
goto_bad;
- return _vg_make_handle(cmd, vg, failure);
+ if (!(vg = _vg_make_handle(cmd, vg, failure)) || vg_read_error(vg))
+ if (!already_locked)
+ unlock_vg(cmd, vg, vg_name);
+
+ return vg;
bad:
if (!already_locked)
@@ -5938,7 +5946,12 @@ struct volume_group *vg_read(struct cmd_context *cmd, const char *vg_name,
struct volume_group *vg_read_for_update(struct cmd_context *cmd, const char *vg_name,
const char *vgid, uint32_t read_flags, uint32_t lockd_state)
{
- return vg_read(cmd, vg_name, vgid, read_flags | READ_FOR_UPDATE, lockd_state);
+ struct volume_group *vg = vg_read(cmd, vg_name, vgid, read_flags | READ_FOR_UPDATE, lockd_state);
+
+ if (!vg || vg_read_error(vg))
+ stack;
+
+ return vg;
}
/*
@@ -5967,9 +5980,8 @@ uint32_t vg_read_error(struct volume_group *vg_handle)
*/
uint32_t vg_lock_newname(struct cmd_context *cmd, const char *vgname)
{
- if (!lock_vol(cmd, vgname, LCK_VG_WRITE, NULL)) {
+ if (!lock_vol(cmd, vgname, LCK_VG_WRITE, NULL))
return FAILED_LOCKING;
- }
/* Find the vgname in the cache */
/* If it's not there we must do full scan to be completely sure */