From be291b330af76d4bfa9aaa9bfd93392554d0a62b Mon Sep 17 00:00:00 2001 From: Mark McLoughlin Date: Thu, 23 Jul 2009 18:31:34 +0100 Subject: [PATCH] Canonicalize qemu machine types In qemu-0.11 there is a 'pc-0.10' machine type which allows you to run guests with a machine which is compatible with the pc machine in qemu-0.10 - e.g. using the original PCI class for virtio-blk and virtio-console and disabling MSI support in virtio-net. The idea here is that we don't want to suprise guests by changing the hardware when qemu is updated. I've just posted some patches for qemu-0.11 which allows libvirt to canonicalize the 'pc' machine alias to the latest machine version. This patches makes us use that so that when a guest is configured to use the 'pc' machine type, we resolve that to 'pc-0.11' machine and save that in the guest XML. See also: https://fedoraproject.org/wiki/Features/KVM_Stable_Guest_ABI * src/qemu_conf.c: add qemudCanonicalizeMachine() to canonicalize the machine type according to the machine aliases in capabilities * src/qemu_driver.c: parse aliases in qemudParseMachineTypesStr() --- src/qemu_conf.c | 11 ++++++- src/qemu_driver.c | 76 +++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 86 insertions(+), 1 deletion(-) diff --git a/src/qemu_conf.c b/src/qemu_conf.c index 581ac8d4b7..598d008d5b 100644 --- a/src/qemu_conf.c +++ b/src/qemu_conf.c @@ -332,7 +332,7 @@ static const struct qemu_arch_info const arch_info_xen[] = { /* Format is: - * [(default)] + * [(default)|(alias of )] */ static int qemudParseMachineTypesStr(const char *output, @@ -380,6 +380,15 @@ qemudParseMachineTypesStr(const char *output, list[0] = machine; nitems++; } + + if ((t = strstr(p, "(alias of ")) && (!next || t < next)) { + p = t + strlen("(alias of "); + if (!(t = strchr(p, ')')) || (next && t >= next)) + continue; + + if (!(machine->canonical = strndup(p, t - p))) + goto error; + } } while ((p = next)); *machines = list; diff --git a/src/qemu_driver.c b/src/qemu_driver.c index c594495cfa..3af35563e9 100644 --- a/src/qemu_driver.c +++ b/src/qemu_driver.c @@ -4257,6 +4257,79 @@ cleanup: return ret; } +static int +qemudCanonicalizeMachineFromInfo(virDomainDefPtr def, + virCapsGuestDomainInfoPtr info, + char **canonical) +{ + int i; + + *canonical = NULL; + + for (i = 0; i < info->nmachines; i++) { + virCapsGuestMachinePtr machine = info->machines[i]; + + if (!machine->canonical) + continue; + + if (strcmp(def->os.machine, machine->name) != 0) + continue; + + if (!(*canonical = strdup(machine->canonical))) { + virReportOOMError(NULL); + return -1; + } + + break; + } + + return 0; +} + +static int +qemudCanonicalizeMachine(virConnectPtr conn, virDomainDefPtr def) +{ + struct qemud_driver *driver = conn->privateData; + char *canonical = NULL; + int i; + + for (i = 0; i < driver->caps->nguests; i++) { + virCapsGuestPtr guest = driver->caps->guests[i]; + int j; + + for (j = 0; j < guest->arch.ndomains; j++) { + virCapsGuestDomainPtr dom = guest->arch.domains[j]; + + if (dom->info.emulator && + STREQ(dom->info.emulator, def->emulator)) { + if (qemudCanonicalizeMachineFromInfo(def, &dom->info, + &canonical) < 0) + return -1; + if (canonical) + goto out; + break; + } + } + + /* if we matched one of the domain's emulators, or if + * we match the default emulator + */ + if (j < guest->arch.ndomains || + (guest->arch.defaultInfo.emulator && + STREQ(guest->arch.defaultInfo.emulator, def->emulator))) { + if (qemudCanonicalizeMachineFromInfo(def, &guest->arch.defaultInfo, + &canonical) < 0) + return -1; + goto out; + } + } +out: + if (canonical) { + VIR_FREE(def->os.machine); + def->os.machine = canonical; + } + return 0; +} static virDomainPtr qemudDomainDefine(virConnectPtr conn, const char *xml) { struct qemud_driver *driver = conn->privateData; @@ -4303,6 +4376,9 @@ static virDomainPtr qemudDomainDefine(virConnectPtr conn, const char *xml) { } } + if (qemudCanonicalizeMachine(conn, def) < 0) + goto cleanup; + if (!(vm = virDomainAssignDef(conn, &driver->domains, def))) {