mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2024-10-30 09:53:10 +00:00
Retain disk PCI address across libvirtd restarts
When we hot-plug a disk device into a qemu guest, we need to retain its PCI address so that it can be removed again later. Currently, we do retain the slot number, but not across libvirtd restarts. Add <state devaddr="xxxx:xx:xx"/> to the disk device XML config when the VIR_DOMAIN_XML_INTERNAL_STATUS flag is used. We still don't parse the domain and bus number, but the format allows us to do that in future. * src/domain_conf.h: replace slotnum with pci_addr struct, add helper for testing whether the address is valid * src/domain_conf.c: handle formatting and parsing the address * src/qemu_driver.c: store the parsed slot number as a full PCI address, and use this address with the pci_del monitor command * src/vbox/vbox_tmpl.c: we're debug printing slotnum here even though it can never be set, just delete it
This commit is contained in:
parent
aa98871c77
commit
0165410712
@ -643,7 +643,7 @@ int virDomainDiskCompare(virDomainDiskDefPtr a,
|
|||||||
static virDomainDiskDefPtr
|
static virDomainDiskDefPtr
|
||||||
virDomainDiskDefParseXML(virConnectPtr conn,
|
virDomainDiskDefParseXML(virConnectPtr conn,
|
||||||
xmlNodePtr node,
|
xmlNodePtr node,
|
||||||
int flags ATTRIBUTE_UNUSED) {
|
int flags) {
|
||||||
virDomainDiskDefPtr def;
|
virDomainDiskDefPtr def;
|
||||||
xmlNodePtr cur;
|
xmlNodePtr cur;
|
||||||
char *type = NULL;
|
char *type = NULL;
|
||||||
@ -654,6 +654,7 @@ virDomainDiskDefParseXML(virConnectPtr conn,
|
|||||||
char *target = NULL;
|
char *target = NULL;
|
||||||
char *bus = NULL;
|
char *bus = NULL;
|
||||||
char *cachetag = NULL;
|
char *cachetag = NULL;
|
||||||
|
char *devaddr = NULL;
|
||||||
|
|
||||||
if (VIR_ALLOC(def) < 0) {
|
if (VIR_ALLOC(def) < 0) {
|
||||||
virReportOOMError(conn);
|
virReportOOMError(conn);
|
||||||
@ -708,6 +709,9 @@ virDomainDiskDefParseXML(virConnectPtr conn,
|
|||||||
def->readonly = 1;
|
def->readonly = 1;
|
||||||
} else if (xmlStrEqual(cur->name, BAD_CAST "shareable")) {
|
} else if (xmlStrEqual(cur->name, BAD_CAST "shareable")) {
|
||||||
def->shared = 1;
|
def->shared = 1;
|
||||||
|
} else if ((flags & VIR_DOMAIN_XML_INTERNAL_STATUS) &&
|
||||||
|
xmlStrEqual(cur->name, BAD_CAST "state")) {
|
||||||
|
devaddr = virXMLPropString(cur, "devaddr");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
cur = cur->next;
|
cur = cur->next;
|
||||||
@ -807,6 +811,17 @@ virDomainDiskDefParseXML(virConnectPtr conn,
|
|||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (devaddr &&
|
||||||
|
sscanf(devaddr, "%x:%x:%x",
|
||||||
|
&def->pci_addr.domain,
|
||||||
|
&def->pci_addr.bus,
|
||||||
|
&def->pci_addr.slot) < 3) {
|
||||||
|
virDomainReportError(conn, VIR_ERR_INTERNAL_ERROR,
|
||||||
|
_("Unable to parse devaddr parameter '%s'"),
|
||||||
|
devaddr);
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
def->src = source;
|
def->src = source;
|
||||||
source = NULL;
|
source = NULL;
|
||||||
def->dst = target;
|
def->dst = target;
|
||||||
@ -825,6 +840,7 @@ cleanup:
|
|||||||
VIR_FREE(driverType);
|
VIR_FREE(driverType);
|
||||||
VIR_FREE(driverName);
|
VIR_FREE(driverName);
|
||||||
VIR_FREE(cachetag);
|
VIR_FREE(cachetag);
|
||||||
|
VIR_FREE(devaddr);
|
||||||
|
|
||||||
return def;
|
return def;
|
||||||
|
|
||||||
@ -3388,7 +3404,8 @@ virDomainLifecycleDefFormat(virConnectPtr conn,
|
|||||||
static int
|
static int
|
||||||
virDomainDiskDefFormat(virConnectPtr conn,
|
virDomainDiskDefFormat(virConnectPtr conn,
|
||||||
virBufferPtr buf,
|
virBufferPtr buf,
|
||||||
virDomainDiskDefPtr def)
|
virDomainDiskDefPtr def,
|
||||||
|
int flags)
|
||||||
{
|
{
|
||||||
const char *type = virDomainDiskTypeToString(def->type);
|
const char *type = virDomainDiskTypeToString(def->type);
|
||||||
const char *device = virDomainDiskDeviceTypeToString(def->device);
|
const char *device = virDomainDiskDeviceTypeToString(def->device);
|
||||||
@ -3446,6 +3463,16 @@ virDomainDiskDefFormat(virConnectPtr conn,
|
|||||||
if (def->shared)
|
if (def->shared)
|
||||||
virBufferAddLit(buf, " <shareable/>\n");
|
virBufferAddLit(buf, " <shareable/>\n");
|
||||||
|
|
||||||
|
if (flags & VIR_DOMAIN_XML_INTERNAL_STATUS) {
|
||||||
|
virBufferAddLit(buf, " <state");
|
||||||
|
if (virDiskHasValidPciAddr(def))
|
||||||
|
virBufferVSprintf(buf, " devaddr='%.4x:%.2x:%.2x'",
|
||||||
|
def->pci_addr.domain,
|
||||||
|
def->pci_addr.bus,
|
||||||
|
def->pci_addr.slot);
|
||||||
|
virBufferAddLit(buf, "/>\n");
|
||||||
|
}
|
||||||
|
|
||||||
virBufferAddLit(buf, " </disk>\n");
|
virBufferAddLit(buf, " </disk>\n");
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -4048,7 +4075,7 @@ char *virDomainDefFormat(virConnectPtr conn,
|
|||||||
def->emulator);
|
def->emulator);
|
||||||
|
|
||||||
for (n = 0 ; n < def->ndisks ; n++)
|
for (n = 0 ; n < def->ndisks ; n++)
|
||||||
if (virDomainDiskDefFormat(conn, &buf, def->disks[n]) < 0)
|
if (virDomainDiskDefFormat(conn, &buf, def->disks[n], flags) < 0)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
||||||
for (n = 0 ; n < def->nfss ; n++)
|
for (n = 0 ; n < def->nfss ; n++)
|
||||||
|
@ -111,9 +111,19 @@ struct _virDomainDiskDef {
|
|||||||
int cachemode;
|
int cachemode;
|
||||||
unsigned int readonly : 1;
|
unsigned int readonly : 1;
|
||||||
unsigned int shared : 1;
|
unsigned int shared : 1;
|
||||||
int slotnum; /* pci slot number for unattach */
|
struct {
|
||||||
|
unsigned domain;
|
||||||
|
unsigned bus;
|
||||||
|
unsigned slot;
|
||||||
|
} pci_addr;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
virDiskHasValidPciAddr(virDomainDiskDefPtr def)
|
||||||
|
{
|
||||||
|
return def->pci_addr.domain || def->pci_addr.domain || def->pci_addr.slot;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Two types of disk backends */
|
/* Two types of disk backends */
|
||||||
enum virDomainFSType {
|
enum virDomainFSType {
|
||||||
|
@ -4390,6 +4390,8 @@ try_command:
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VIR_FREE(cmd);
|
||||||
|
|
||||||
DEBUG ("%s: pci_add reply: %s", vm->def->name, reply);
|
DEBUG ("%s: pci_add reply: %s", vm->def->name, reply);
|
||||||
/* If the command succeeds qemu prints:
|
/* If the command succeeds qemu prints:
|
||||||
* OK bus 0, slot XXX...
|
* OK bus 0, slot XXX...
|
||||||
@ -4399,22 +4401,25 @@ try_command:
|
|||||||
if ((s = strstr(reply, "OK ")) &&
|
if ((s = strstr(reply, "OK ")) &&
|
||||||
(s = strstr(s, "slot "))) {
|
(s = strstr(s, "slot "))) {
|
||||||
char *dummy = s;
|
char *dummy = s;
|
||||||
|
unsigned slot;
|
||||||
|
|
||||||
s += strlen("slot ");
|
s += strlen("slot ");
|
||||||
|
|
||||||
if (virStrToLong_i ((const char*)s, &dummy, 10, &dev->data.disk->slotnum) == -1)
|
if (virStrToLong_ui((const char*)s, &dummy, 10, &slot) == -1)
|
||||||
VIR_WARN("%s", _("Unable to parse slot number\n"));
|
VIR_WARN("%s", _("Unable to parse slot number\n"));
|
||||||
|
|
||||||
/* XXX not neccessarily always going to end up in domain 0 / bus 0 :-( */
|
/* XXX not neccessarily always going to end up in domain 0 / bus 0 :-( */
|
||||||
/* XXX this slotnum is not persistant across restarts :-( */
|
dev->data.disk->pci_addr.domain = 0;
|
||||||
|
dev->data.disk->pci_addr.bus = 0;
|
||||||
|
dev->data.disk->pci_addr.slot = slot;
|
||||||
} else if (!tryOldSyntax && strstr(reply, "invalid char in expression")) {
|
} else if (!tryOldSyntax && strstr(reply, "invalid char in expression")) {
|
||||||
VIR_FREE(reply);
|
VIR_FREE(reply);
|
||||||
VIR_FREE(cmd);
|
|
||||||
tryOldSyntax = 1;
|
tryOldSyntax = 1;
|
||||||
goto try_command;
|
goto try_command;
|
||||||
} else {
|
} else {
|
||||||
qemudReportError (conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
|
qemudReportError (conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
|
||||||
_("adding %s disk failed: %s"), type, reply);
|
_("adding %s disk failed: %s"), type, reply);
|
||||||
VIR_FREE(reply);
|
VIR_FREE(reply);
|
||||||
VIR_FREE(cmd);
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4423,7 +4428,7 @@ try_command:
|
|||||||
virDomainDiskQSort);
|
virDomainDiskQSort);
|
||||||
|
|
||||||
VIR_FREE(reply);
|
VIR_FREE(reply);
|
||||||
VIR_FREE(cmd);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4660,21 +4665,24 @@ static int qemudDomainDetachPciDiskDevice(virConnectPtr conn,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (detach->slotnum < 1) {
|
if (!virDiskHasValidPciAddr(detach)) {
|
||||||
qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
|
qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
|
||||||
_("disk %s cannot be detached - invalid slot number %d"),
|
_("disk %s cannot be detached - no PCI address for device"),
|
||||||
detach->dst, detach->slotnum);
|
detach->dst);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
try_command:
|
try_command:
|
||||||
if (tryOldSyntax) {
|
if (tryOldSyntax) {
|
||||||
if (virAsprintf(&cmd, "pci_del 0 %d", detach->slotnum) < 0) {
|
if (virAsprintf(&cmd, "pci_del 0 %.2x", detach->pci_addr.slot) < 0) {
|
||||||
virReportOOMError(conn);
|
virReportOOMError(conn);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (virAsprintf(&cmd, "pci_del pci_addr=0:0:%d", detach->slotnum) < 0) {
|
if (virAsprintf(&cmd, "pci_del pci_addr=%.4x:%.2x:%.2x",
|
||||||
|
detach->pci_addr.domain,
|
||||||
|
detach->pci_addr.bus,
|
||||||
|
detach->pci_addr.slot) < 0) {
|
||||||
virReportOOMError(conn);
|
virReportOOMError(conn);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
@ -4698,8 +4706,12 @@ try_command:
|
|||||||
if (strstr(reply, "invalid slot") ||
|
if (strstr(reply, "invalid slot") ||
|
||||||
strstr(reply, "Invalid pci address")) {
|
strstr(reply, "Invalid pci address")) {
|
||||||
qemudReportError (conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
|
qemudReportError (conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
|
||||||
_("failed to detach disk %s: invalid slot %d: %s"),
|
_("failed to detach disk %s: invalid PCI address %.4x:%.2x:%.2x: %s"),
|
||||||
detach->dst, detach->slotnum, reply);
|
detach->dst,
|
||||||
|
detach->pci_addr.domain,
|
||||||
|
detach->pci_addr.bus,
|
||||||
|
detach->pci_addr.slot,
|
||||||
|
reply);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2712,7 +2712,6 @@ static virDomainPtr vboxDomainDefineXML(virConnectPtr conn, const char *xml) {
|
|||||||
DEBUG("disk(%d) cachemode: %d", i, def->disks[i]->cachemode);
|
DEBUG("disk(%d) cachemode: %d", i, def->disks[i]->cachemode);
|
||||||
DEBUG("disk(%d) readonly: %s", i, def->disks[i]->readonly ? "True" : "False");
|
DEBUG("disk(%d) readonly: %s", i, def->disks[i]->readonly ? "True" : "False");
|
||||||
DEBUG("disk(%d) shared: %s", i, def->disks[i]->shared ? "True" : "False");
|
DEBUG("disk(%d) shared: %s", i, def->disks[i]->shared ? "True" : "False");
|
||||||
DEBUG("disk(%d) slotnum: %d", i, def->disks[i]->slotnum);
|
|
||||||
|
|
||||||
if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
||||||
if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_FILE) {
|
if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_FILE) {
|
||||||
|
Loading…
Reference in New Issue
Block a user