[libvirt] [PATCH 1.2.8] storage: zfs: fix double listing of new volumes

Roman Bogorodskiy bogorodskiy at gmail.com
Wed Aug 27 09:02:04 UTC 2014


Currently, after calling commands to create a new volumes,
virStorageBackendZFSCreateVol calls virStorageBackendZFSFindVols that
calls virStorageBackendZFSParseVol.

virStorageBackendZFSParseVol checks if a volume already exists by
trying to get it using virStorageVolDefFindByName.

For a just created volume it returns NULL, so volume is reported as
new and appended to pool->volumes. This causes a volume to be listed
twice as storageVolCreateXML appends this new volume to the list as
well.

Fix that by passing a new volume definition to
virStorageBackendZFSParseVol so it could determine if it needs to add
this volume to the list.
---
 src/storage/storage_backend_zfs.c | 45 ++++++++++++++++++++++-----------------
 1 file changed, 26 insertions(+), 19 deletions(-)

diff --git a/src/storage/storage_backend_zfs.c b/src/storage/storage_backend_zfs.c
index 2d74055..d8201ac 100644
--- a/src/storage/storage_backend_zfs.c
+++ b/src/storage/storage_backend_zfs.c
@@ -58,7 +58,8 @@ virStorageBackendZFSCheckPool(virConnectPtr conn ATTRIBUTE_UNUSED,
 
 static int
 virStorageBackendZFSParseVol(virStoragePoolObjPtr pool,
-                             const char *volume)
+                             virStorageVolDefPtr vol,
+                             const char *volume_string)
 {
     int ret = -1;
     char **tokens;
@@ -66,9 +67,9 @@ virStorageBackendZFSParseVol(virStoragePoolObjPtr pool,
     char **name_tokens = NULL;
     char *vol_name;
     bool is_new_vol = false;
-    virStorageVolDefPtr vol = NULL;
+    virStorageVolDefPtr volume = NULL;
 
-    if (!(tokens = virStringSplitCount(volume, "\t", 0, &count)))
+    if (!(tokens = virStringSplitCount(volume_string, "\t", 0, &count)))
         return -1;
 
     if (count != 2)
@@ -79,36 +80,41 @@ virStorageBackendZFSParseVol(virStoragePoolObjPtr pool,
 
     vol_name = name_tokens[1];
 
-    vol = virStorageVolDefFindByName(pool, vol_name);
+    if (vol == NULL)
+        volume = virStorageVolDefFindByName(pool, vol_name);
+    else
+        volume = vol;
 
-    if (vol == NULL) {
-        if (VIR_ALLOC(vol) < 0)
+    if (volume == NULL) {
+        if (VIR_ALLOC(volume) < 0)
             goto cleanup;
 
         is_new_vol = true;
-        vol->type = VIR_STORAGE_VOL_BLOCK;
+        volume->type = VIR_STORAGE_VOL_BLOCK;
 
-        if (VIR_STRDUP(vol->name, vol_name) < 0)
+        if (VIR_STRDUP(volume->name, vol_name) < 0)
             goto cleanup;
     }
 
-    if (!vol->key && VIR_STRDUP(vol->key, tokens[0]) < 0)
+    if (!volume->key && VIR_STRDUP(volume->key, tokens[0]) < 0)
         goto cleanup;
 
-    if (vol->target.path == NULL) {
-        if (virAsprintf(&vol->target.path, "%s/%s",
-                        pool->def->target.path, vol->name) < 0)
+    if (volume->target.path == NULL) {
+        if (virAsprintf(&volume->target.path, "%s/%s",
+                        pool->def->target.path, volume->name) < 0)
             goto cleanup;
     }
 
-    if (virStrToLong_ull(tokens[1], NULL, 10, &vol->target.capacity) < 0) {
+    if (virStrToLong_ull(tokens[1], NULL, 10, &volume->target.capacity) < 0) {
         virReportError(VIR_ERR_INTERNAL_ERROR,
                        "%s", _("malformed volsize reported"));
         goto cleanup;
     }
 
     if (is_new_vol &&
-        VIR_APPEND_ELEMENT(pool->volumes.objs, pool->volumes.count, vol) < 0)
+        VIR_APPEND_ELEMENT(pool->volumes.objs,
+                           pool->volumes.count,
+                           volume) < 0)
         goto cleanup;
 
     ret = 0;
@@ -116,12 +122,13 @@ virStorageBackendZFSParseVol(virStoragePoolObjPtr pool,
     virStringFreeList(tokens);
     virStringFreeList(name_tokens);
     if (is_new_vol && (ret == -1))
-        virStorageVolDefFree(vol);
+        virStorageVolDefFree(volume);
     return ret;
 }
 
 static int
-virStorageBackendZFSFindVols(virStoragePoolObjPtr pool)
+virStorageBackendZFSFindVols(virStoragePoolObjPtr pool,
+                             virStorageVolDefPtr vol)
 {
     virCommandPtr cmd = NULL;
     char *volumes_list = NULL;
@@ -157,7 +164,7 @@ virStorageBackendZFSFindVols(virStoragePoolObjPtr pool)
         if (STREQ(lines[i], ""))
             continue;
 
-        if (virStorageBackendZFSParseVol(pool, lines[i]) < 0)
+        if (virStorageBackendZFSParseVol(pool, vol, lines[i]) < 0)
             continue;
     }
 
@@ -233,7 +240,7 @@ virStorageBackendZFSRefreshPool(virConnectPtr conn ATTRIBUTE_UNUSED,
     }
 
     /* Obtain a list of volumes */
-    if (virStorageBackendZFSFindVols(pool) < 0)
+    if (virStorageBackendZFSFindVols(pool, NULL) < 0)
         goto cleanup;
 
  cleanup:
@@ -285,7 +292,7 @@ virStorageBackendZFSCreateVol(virConnectPtr conn ATTRIBUTE_UNUSED,
     if (virCommandRun(cmd, NULL) < 0)
         goto cleanup;
 
-    if (virStorageBackendZFSFindVols(pool) < 0)
+    if (virStorageBackendZFSFindVols(pool, vol) < 0)
         goto cleanup;
 
     ret = 0;
-- 
2.0.2




More information about the libvir-list mailing list