[PATCH RFC 36/51] qemu: domain: Store blockjob data in the status XML

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



We need to store the block job state in the status XML so that we can
properly recover any data when reconnecting after startup and also in
the end to be able to do any transition of the backing chain that
happened while libvirt was not connected to the monitor.

First step is to note the name, type, state and corresponding disk into
the status XML.

Signed-off-by: Peter Krempa <pkrempa@xxxxxxxxxx>
---
 src/qemu/qemu_domain.c | 105 +++++++++++++++++++++++++++++++++++++++--
 1 file changed, 101 insertions(+), 4 deletions(-)

diff --git a/src/qemu/qemu_domain.c b/src/qemu/qemu_domain.c
index 7a9a94efcf..7845889dec 100644
--- a/src/qemu/qemu_domain.c
+++ b/src/qemu/qemu_domain.c
@@ -35,6 +35,7 @@
 #include "qemu_migration_params.h"
 #include "qemu_security.h"
 #include "qemu_extdevice.h"
+#include "qemu_blockjob.h"
 #include "viralloc.h"
 #include "virlog.h"
 #include "virerror.h"
@@ -2235,17 +2236,47 @@ qemuDomainObjPrivateXMLFormatAutomaticPlacement(virBufferPtr buf,
 }


+static int
+qemuDomainObjPrivateXMLFormatBlockjobIterator(void *payload,
+                                              const void *name ATTRIBUTE_UNUSED,
+                                              void *data)
+{
+    virBuffer attrBuf = VIR_BUFFER_INITIALIZER;
+    virBuffer childBuf = VIR_BUFFER_INITIALIZER;
+    qemuBlockJobDataPtr job = payload;
+    virBufferPtr buf = data;
+
+    virBufferSetChildIndent(&childBuf, buf);
+
+    virBufferAsprintf(&attrBuf, " name='%s'", job->name);
+    virBufferAsprintf(&attrBuf, " type='%s'", qemuBlockjobTypeToString(job->type));
+    virBufferAsprintf(&attrBuf, " state='%s'", qemuBlockjobStateTypeToString(job->state));
+
+    if (job->disk)
+        virBufferAsprintf(&childBuf, "<disk dst='%s'/>\n", job->disk->dst);
+
+    return virXMLFormatElement(buf, "blockjob", &attrBuf, &childBuf);
+}
+
+
 static int
 qemuDomainObjPrivateXMLFormatBlockjobs(virBufferPtr buf,
                                        virDomainObjPtr vm)
 {
+    qemuDomainObjPrivatePtr priv = vm->privateData;
     virBuffer attrBuf = VIR_BUFFER_INITIALIZER;
+    virBuffer childBuf = VIR_BUFFER_INITIALIZER;
     bool bj = qemuDomainHasBlockjob(vm, false);

     virBufferAsprintf(&attrBuf, " active='%s'",
                       virTristateBoolTypeToString(virTristateBoolFromBool(bj)));

-    return virXMLFormatElement(buf, "blockjobs", &attrBuf, NULL);
+    virBufferSetChildIndent(&childBuf, buf);
+
+    virHashForEach(priv->blockjobs,
+                   qemuDomainObjPrivateXMLFormatBlockjobIterator, &childBuf);
+
+    return virXMLFormatElement(buf, "blockjobs", &attrBuf, &childBuf);
 }


@@ -2599,18 +2630,84 @@ qemuDomainObjPrivateXMLParseAutomaticPlacement(xmlXPathContextPtr ctxt,


 static int
-qemuDomainObjPrivateXMLParseBlockjobs(qemuDomainObjPrivatePtr priv,
+qemuDomainObjPrivateXMLParseBlockjobData(virDomainObjPtr vm,
+                                         xmlNodePtr node,
+                                         xmlXPathContextPtr ctxt)
+{
+    xmlNodePtr save_node = ctxt->node;
+    virDomainDiskDefPtr disk = NULL;
+    qemuBlockJobDataPtr job = NULL;
+    char *name = NULL;
+    char *typestr = NULL;
+    int type;
+    char *statestr = NULL;
+    int state;
+    char *diskdst = NULL;
+    int ret = -1;
+
+    ctxt->node = node;
+
+    if (!(name = virXPathString("string(./@name)", ctxt)) ||
+        !(typestr = virXPathString("string(./@type)", ctxt)) ||
+        !(statestr = virXPathString("string(./@state)", ctxt)) ||
+        (type = qemuBlockjobTypeFromString(typestr)) < 0 ||
+        (state = qemuBlockjobStateTypeFromString(statestr)) < 0) {
+        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
+                       _("malformed block job data"));
+        goto cleanup;
+    }
+
+    if ((diskdst = virXPathString("string(./disk/@dst)", ctxt)))
+        disk = virDomainDiskByName(vm->def, diskdst, false);
+
+    if (!(job = qemuBlockJobDiskNew(vm, disk, type, name)))
+        goto cleanup;
+
+    job->state = state;
+
+    ret = 0;
+
+ cleanup:
+    virObjectUnref(job);
+    VIR_FREE(name);
+    VIR_FREE(typestr);
+    VIR_FREE(statestr);
+    VIR_FREE(diskdst);
+    ctxt->node = save_node;
+    return ret;
+}
+
+
+static int
+qemuDomainObjPrivateXMLParseBlockjobs(virDomainObjPtr vm,
+                                      qemuDomainObjPrivatePtr priv,
                                       xmlXPathContextPtr ctxt)
 {
+    xmlNodePtr *nodes = NULL;
+    ssize_t nnodes = 0;
     char *active;
     int tmp;
+    size_t i;
+    int ret = -1;

     if ((active = virXPathString("string(./blockjobs/@active)", ctxt)) &&
         (tmp = virTristateBoolTypeFromString(active)) > 0)
         priv->reconnectBlockjobs = tmp;

+    if ((nnodes = virXPathNodeSet("./blockjobs/blockjob", ctxt, &nodes)) < 0)
+        goto cleanup;
+
+    for (i = 0; i < nnodes; i++) {
+        if (qemuDomainObjPrivateXMLParseBlockjobData(vm, nodes[i], ctxt) < 0)
+            goto cleanup;
+    }
+
+    ret = 0;
+
+ cleanup:
     VIR_FREE(active);
-    return 0;
+    VIR_FREE(nodes);
+    return ret;
 }


@@ -2989,7 +3086,7 @@ qemuDomainObjPrivateXMLParse(xmlXPathContextPtr ctxt,

     qemuDomainObjPrivateXMLParsePR(ctxt, &priv->prDaemonRunning);

-    if (qemuDomainObjPrivateXMLParseBlockjobs(priv, ctxt) < 0)
+    if (qemuDomainObjPrivateXMLParseBlockjobs(vm, priv, ctxt) < 0)
         goto error;

     qemuDomainStorageIdReset(priv);
-- 
2.19.2

--
libvir-list mailing list
libvir-list@xxxxxxxxxx
https://www.redhat.com/mailman/listinfo/libvir-list



[Index of Archives]     [Virt Tools]     [Libvirt Users]     [Lib OS Info]     [Fedora Users]     [Fedora Desktop]     [Fedora SELinux]     [Big List of Linux Books]     [Yosemite News]     [KDE Users]     [Fedora Tools]

  Powered by Linux