/* * domain_capabilities.c: domain capabilities XML processing * * Copyright (C) 2014 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see * . * * Author: Michal Privoznik */ #include #include "domain_capabilities.h" #include "domain_conf.h" #include "viralloc.h" #include "virstring.h" #define VIR_FROM_THIS VIR_FROM_CAPABILITIES VIR_ENUM_IMPL(virDomainCapsCPUUsable, VIR_DOMCAPS_CPU_USABLE_LAST, "unknown", "yes", "no"); static virClassPtr virDomainCapsClass; static virClassPtr virDomainCapsCPUModelsClass; static void virDomainCapsDispose(void *obj); static void virDomainCapsCPUModelsDispose(void *obj); static int virDomainCapsOnceInit(void) { if (!VIR_CLASS_NEW(virDomainCaps, virClassForObjectLockable())) return -1; if (!VIR_CLASS_NEW(virDomainCapsCPUModels, virClassForObject())) return -1; return 0; } VIR_ONCE_GLOBAL_INIT(virDomainCaps) static void virDomainCapsStringValuesFree(virDomainCapsStringValuesPtr values) { size_t i; if (!values || !values->values) return; for (i = 0; i < values->nvalues; i++) VIR_FREE(values->values[i]); VIR_FREE(values->values); } void virSEVCapabilitiesFree(virSEVCapability *cap) { if (!cap) return; VIR_FREE(cap->pdh); VIR_FREE(cap->cert_chain); VIR_FREE(cap); } static void virDomainCapsDispose(void *obj) { virDomainCapsPtr caps = obj; VIR_FREE(caps->path); VIR_FREE(caps->machine); virObjectUnref(caps->cpu.custom); virCPUDefFree(caps->cpu.hostModel); virSEVCapabilitiesFree(caps->sev); virDomainCapsStringValuesFree(&caps->os.loader.values); } static void virDomainCapsCPUModelsDispose(void *obj) { virDomainCapsCPUModelsPtr cpuModels = obj; size_t i; for (i = 0; i < cpuModels->nmodels; i++) { VIR_FREE(cpuModels->models[i].name); virStringListFree(cpuModels->models[i].blockers); } VIR_FREE(cpuModels->models); } virDomainCapsPtr virDomainCapsNew(const char *path, const char *machine, virArch arch, virDomainVirtType virttype) { virDomainCapsPtr caps = NULL; if (virDomainCapsInitialize() < 0) return NULL; if (!(caps = virObjectLockableNew(virDomainCapsClass))) return NULL; if (VIR_STRDUP(caps->path, path) < 0 || VIR_STRDUP(caps->machine, machine) < 0) goto error; caps->arch = arch; caps->virttype = virttype; return caps; error: virObjectUnref(caps); return NULL; } virDomainCapsCPUModelsPtr virDomainCapsCPUModelsNew(size_t nmodels) { virDomainCapsCPUModelsPtr cpuModels = NULL; if (virDomainCapsInitialize() < 0) return NULL; if (!(cpuModels = virObjectNew(virDomainCapsCPUModelsClass))) return NULL; if (VIR_ALLOC_N(cpuModels->models, nmodels) < 0) goto error; cpuModels->nmodels_max = nmodels; return cpuModels; error: virObjectUnref(cpuModels); return NULL; } virDomainCapsCPUModelsPtr virDomainCapsCPUModelsCopy(virDomainCapsCPUModelsPtr old) { virDomainCapsCPUModelsPtr cpuModels; size_t i; if (!(cpuModels = virDomainCapsCPUModelsNew(old->nmodels))) return NULL; for (i = 0; i < old->nmodels; i++) { if (virDomainCapsCPUModelsAdd(cpuModels, old->models[i].name, -1, old->models[i].usable, old->models[i].blockers) < 0) goto error; } return cpuModels; error: virObjectUnref(cpuModels); return NULL; } virDomainCapsCPUModelsPtr virDomainCapsCPUModelsFilter(virDomainCapsCPUModelsPtr old, const char **models, const char **blacklist) { virDomainCapsCPUModelsPtr cpuModels; size_t i; if (!(cpuModels = virDomainCapsCPUModelsNew(0))) return NULL; for (i = 0; i < old->nmodels; i++) { if (models && !virStringListHasString(models, old->models[i].name)) continue; if (blacklist && virStringListHasString(blacklist, old->models[i].name)) continue; if (virDomainCapsCPUModelsAdd(cpuModels, old->models[i].name, -1, old->models[i].usable, old->models[i].blockers) < 0) goto error; } return cpuModels; error: virObjectUnref(cpuModels); return NULL; } int virDomainCapsCPUModelsAddSteal(virDomainCapsCPUModelsPtr cpuModels, char **name, virDomainCapsCPUUsable usable, char ***blockers) { if (VIR_RESIZE_N(cpuModels->models, cpuModels->nmodels_max, cpuModels->nmodels, 1) < 0) return -1; cpuModels->models[cpuModels->nmodels].usable = usable; VIR_STEAL_PTR(cpuModels->models[cpuModels->nmodels].name, *name); if (blockers) VIR_STEAL_PTR(cpuModels->models[cpuModels->nmodels].blockers, *blockers); cpuModels->nmodels++; return 0; } int virDomainCapsCPUModelsAdd(virDomainCapsCPUModelsPtr cpuModels, const char *name, ssize_t nameLen, virDomainCapsCPUUsable usable, char **blockers) { char *nameCopy = NULL; char **blockersCopy = NULL; if (VIR_STRNDUP(nameCopy, name, nameLen) < 0) goto error; if (virStringListCopy(&blockersCopy, (const char **)blockers) < 0) goto error; if (virDomainCapsCPUModelsAddSteal(cpuModels, &nameCopy, usable, &blockersCopy) < 0) goto error; return 0; error: VIR_FREE(nameCopy); virStringListFree(blockersCopy); return -1; } virDomainCapsCPUModelPtr virDomainCapsCPUModelsGet(virDomainCapsCPUModelsPtr cpuModels, const char *name) { size_t i; if (!cpuModels) return NULL; for (i = 0; i < cpuModels->nmodels; i++) { if (STRCASEEQ(cpuModels->models[i].name, name)) return cpuModels->models + i; } return NULL; } int virDomainCapsEnumSet(virDomainCapsEnumPtr capsEnum, const char *capsEnumName, size_t nvalues, unsigned int *values) { int ret = -1; size_t i; for (i = 0; i < nvalues; i++) { unsigned int val = 1 << values[i]; if (!val) { /* Integer overflow */ virReportError(VIR_ERR_INTERNAL_ERROR, _("integer overflow on %s. Please contact the " "libvirt development team at libvir-list@redhat.com"), capsEnumName); goto cleanup; } capsEnum->values |= val; } ret = 0; cleanup: return ret; } void virDomainCapsEnumClear(virDomainCapsEnumPtr capsEnum) { capsEnum->values = 0; } static int virDomainCapsEnumFormat(virBufferPtr buf, virDomainCapsEnumPtr capsEnum, const char *capsEnumName, virDomainCapsValToStr valToStr) { int ret = -1; size_t i; virBufferAsprintf(buf, "values) { virBufferAddLit(buf, "/>\n"); ret = 0; goto cleanup; } virBufferAddLit(buf, ">\n"); virBufferAdjustIndent(buf, 2); for (i = 0; i < sizeof(capsEnum->values) * CHAR_BIT; i++) { const char *val; if (!(capsEnum->values & (1 << i))) continue; if ((val = (valToStr)(i))) virBufferAsprintf(buf, "%s\n", val); } virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); ret = 0; cleanup: return ret; } static void virDomainCapsStringValuesFormat(virBufferPtr buf, virDomainCapsStringValuesPtr values) { size_t i; for (i = 0; i < values->nvalues; i++) virBufferEscapeString(buf, "%s\n", values->values[i]); } #define FORMAT_PROLOGUE(item) \ do { \ virBufferAsprintf(buf, "<" #item " supported='%s'%s\n", \ item->supported ? "yes" : "no", \ item->supported ? ">" : "/>"); \ if (!item->supported) \ return; \ virBufferAdjustIndent(buf, 2); \ } while (0) #define FORMAT_EPILOGUE(item) \ do { \ virBufferAdjustIndent(buf, -2); \ virBufferAddLit(buf, "\n"); \ } while (0) #define ENUM_PROCESS(master, capsEnum, valToStr) \ do { \ virDomainCapsEnumFormat(buf, &master->capsEnum, \ #capsEnum, valToStr); \ } while (0) static void virDomainCapsLoaderFormat(virBufferPtr buf, virDomainCapsLoaderPtr loader) { FORMAT_PROLOGUE(loader); virDomainCapsStringValuesFormat(buf, &loader->values); ENUM_PROCESS(loader, type, virDomainLoaderTypeToString); ENUM_PROCESS(loader, readonly, virTristateBoolTypeToString); FORMAT_EPILOGUE(loader); } static void virDomainCapsOSFormat(virBufferPtr buf, virDomainCapsOSPtr os) { virDomainCapsLoaderPtr loader = &os->loader; FORMAT_PROLOGUE(os); virDomainCapsLoaderFormat(buf, loader); FORMAT_EPILOGUE(os); } static void virDomainCapsCPUCustomFormat(virBufferPtr buf, virDomainCapsCPUModelsPtr custom) { size_t i; virBufferAdjustIndent(buf, 2); for (i = 0; i < custom->nmodels; i++) { virDomainCapsCPUModelPtr model = custom->models + i; virBufferAsprintf(buf, "%s\n", virDomainCapsCPUUsableTypeToString(model->usable), model->name); } virBufferAdjustIndent(buf, -2); } static void virDomainCapsCPUFormat(virBufferPtr buf, virDomainCapsCPUPtr cpu) { virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferAsprintf(buf, "\n", virCPUModeTypeToString(VIR_CPU_MODE_HOST_PASSTHROUGH), cpu->hostPassthrough ? "yes" : "no"); virBufferAsprintf(buf, "hostModel) { virBufferAddLit(buf, "supported='yes'>\n"); virBufferAdjustIndent(buf, 2); virCPUDefFormatBuf(buf, cpu->hostModel); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } else { virBufferAddLit(buf, "supported='no'/>\n"); } virBufferAsprintf(buf, "custom && cpu->custom->nmodels) { virBufferAddLit(buf, "supported='yes'>\n"); virDomainCapsCPUCustomFormat(buf, cpu->custom); virBufferAddLit(buf, "\n"); } else { virBufferAddLit(buf, "supported='no'/>\n"); } virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } static void virDomainCapsDeviceDiskFormat(virBufferPtr buf, virDomainCapsDeviceDiskPtr const disk) { FORMAT_PROLOGUE(disk); ENUM_PROCESS(disk, diskDevice, virDomainDiskDeviceTypeToString); ENUM_PROCESS(disk, bus, virDomainDiskBusTypeToString); FORMAT_EPILOGUE(disk); } static void virDomainCapsDeviceGraphicsFormat(virBufferPtr buf, virDomainCapsDeviceGraphicsPtr const graphics) { FORMAT_PROLOGUE(graphics); ENUM_PROCESS(graphics, type, virDomainGraphicsTypeToString); FORMAT_EPILOGUE(graphics); } static void virDomainCapsDeviceVideoFormat(virBufferPtr buf, virDomainCapsDeviceVideoPtr const video) { FORMAT_PROLOGUE(video); ENUM_PROCESS(video, modelType, virDomainVideoTypeToString); FORMAT_EPILOGUE(video); } static void virDomainCapsDeviceHostdevFormat(virBufferPtr buf, virDomainCapsDeviceHostdevPtr const hostdev) { FORMAT_PROLOGUE(hostdev); ENUM_PROCESS(hostdev, mode, virDomainHostdevModeTypeToString); ENUM_PROCESS(hostdev, startupPolicy, virDomainStartupPolicyTypeToString); ENUM_PROCESS(hostdev, subsysType, virDomainHostdevSubsysTypeToString); ENUM_PROCESS(hostdev, capsType, virDomainHostdevCapsTypeToString); ENUM_PROCESS(hostdev, pciBackend, virDomainHostdevSubsysPCIBackendTypeToString); FORMAT_EPILOGUE(hostdev); } /** * virDomainCapsFeatureGICFormat: * @buf: target buffer * @gic: GIC features * * Format GIC features for inclusion in the domcapabilities XML. * * The resulting XML will look like * * * \n"); } else { virBufferAddLit(buf, "\n"); virBufferAdjustIndent(buf, 2); virBufferAsprintf(buf, "%d\n", sev->cbitpos); virBufferAsprintf(buf, "%d\n", sev->reduced_phys_bits); virBufferAdjustIndent(buf, -2); virBufferAddLit(buf, "\n"); } return; } char * virDomainCapsFormat(virDomainCapsPtr const caps) { virBuffer buf = VIR_BUFFER_INITIALIZER; const char *virttype_str = virDomainVirtTypeToString(caps->virttype); const char *arch_str = virArchToString(caps->arch); virBufferAddLit(&buf, "\n"); virBufferAdjustIndent(&buf, 2); virBufferEscapeString(&buf, "%s\n", caps->path); virBufferAsprintf(&buf, "%s\n", virttype_str); if (caps->machine) virBufferAsprintf(&buf, "%s\n", caps->machine); virBufferAsprintf(&buf, "%s\n", arch_str); if (caps->maxvcpus) virBufferAsprintf(&buf, "\n", caps->maxvcpus); virBufferAsprintf(&buf, "\n", caps->iothreads ? "yes" : "no"); virDomainCapsOSFormat(&buf, &caps->os); virDomainCapsCPUFormat(&buf, &caps->cpu); virBufferAddLit(&buf, "\n"); virBufferAdjustIndent(&buf, 2); virDomainCapsDeviceDiskFormat(&buf, &caps->disk); virDomainCapsDeviceGraphicsFormat(&buf, &caps->graphics); virDomainCapsDeviceVideoFormat(&buf, &caps->video); virDomainCapsDeviceHostdevFormat(&buf, &caps->hostdev); virBufferAdjustIndent(&buf, -2); virBufferAddLit(&buf, "\n"); virBufferAddLit(&buf, "\n"); virBufferAdjustIndent(&buf, 2); virDomainCapsFeatureGICFormat(&buf, &caps->gic); virBufferAsprintf(&buf, "\n", caps->vmcoreinfo ? "yes" : "no"); virBufferAsprintf(&buf, "\n", caps->genid ? "yes" : "no"); virDomainCapsFeatureSEVFormat(&buf, caps->sev); virBufferAdjustIndent(&buf, -2); virBufferAddLit(&buf, "\n"); virBufferAdjustIndent(&buf, -2); virBufferAddLit(&buf, "\n"); virBufferCheckError(&buf); return virBufferContentAndReset(&buf); }