libxl: vnuma support

This patch generates a NUMA distance-aware libxl description from the
information extracted from a NUMA distance-aware libvirt XML file.

By default, if no NUMA node distance information is supplied in the
libvirt XML file, this patch uses the distances 10 for local and 20
for remote nodes/sockets.

Signed-off-by: Wim ten Have <wim.ten.have@oracle.com>
Reviewed-by: Jim Fehlig <jfehlig@suse.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
This commit is contained in:
Wim ten Have 2017-11-02 16:47:22 +01:00 committed by Jim Fehlig
parent 03d0959af3
commit c9a5682ffa
2 changed files with 125 additions and 1 deletions

View File

@ -593,6 +593,120 @@ libxlMakeDomBuildInfo(virDomainDefPtr def,
return 0; return 0;
} }
#ifdef LIBXL_HAVE_VNUMA
static int
libxlMakeVnumaList(virDomainDefPtr def,
libxl_ctx *ctx,
libxl_domain_config *d_config)
{
int ret = -1;
size_t i, j;
size_t nr_nodes;
size_t num_vnuma;
bool simulate = false;
virBitmapPtr bitmap = NULL;
virDomainNumaPtr numa = def->numa;
libxl_domain_build_info *b_info = &d_config->b_info;
libxl_physinfo physinfo;
libxl_vnode_info *vnuma_nodes = NULL;
if (!numa)
return 0;
num_vnuma = virDomainNumaGetNodeCount(numa);
if (!num_vnuma)
return 0;
libxl_physinfo_init(&physinfo);
if (libxl_get_physinfo(ctx, &physinfo) < 0) {
libxl_physinfo_dispose(&physinfo);
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
_("libxl_get_physinfo_info failed"));
return -1;
}
nr_nodes = physinfo.nr_nodes;
libxl_physinfo_dispose(&physinfo);
if (num_vnuma > nr_nodes) {
VIR_WARN("Number of configured numa cells %zu exceeds available physical nodes %zu. All cells will use physical node 0",
num_vnuma, nr_nodes);
simulate = true;
}
/*
* allocate the vnuma_nodes for assignment under b_info.
*/
if (VIR_ALLOC_N(vnuma_nodes, num_vnuma) < 0)
return -1;
/*
* parse the vnuma vnodes data.
*/
for (i = 0; i < num_vnuma; i++) {
int cpu;
libxl_bitmap vcpu_bitmap;
libxl_vnode_info *p = &vnuma_nodes[i];
libxl_vnode_info_init(p);
/* pnode */
p->pnode = simulate ? 0 : i;
/* memory size */
p->memkb = virDomainNumaGetNodeMemorySize(numa, i);
/* vcpus */
bitmap = virDomainNumaGetNodeCpumask(numa, i);
if (bitmap == NULL) {
virReportError(VIR_ERR_INTERNAL_ERROR,
_("vnuma sibling %zu missing vcpus set"), i);
goto cleanup;
}
if ((cpu = virBitmapNextSetBit(bitmap, -1)) < 0)
goto cleanup;
libxl_bitmap_init(&vcpu_bitmap);
if (libxl_cpu_bitmap_alloc(ctx, &vcpu_bitmap, b_info->max_vcpus)) {
virReportOOMError();
goto cleanup;
}
do {
libxl_bitmap_set(&vcpu_bitmap, cpu);
} while ((cpu = virBitmapNextSetBit(bitmap, cpu)) >= 0);
libxl_bitmap_copy_alloc(ctx, &p->vcpus, &vcpu_bitmap);
libxl_bitmap_dispose(&vcpu_bitmap);
/* vdistances */
if (VIR_ALLOC_N(p->distances, num_vnuma) < 0)
goto cleanup;
p->num_distances = num_vnuma;
for (j = 0; j < num_vnuma; j++)
p->distances[j] = virDomainNumaGetNodeDistance(numa, i, j);
}
b_info->vnuma_nodes = vnuma_nodes;
b_info->num_vnuma_nodes = num_vnuma;
ret = 0;
cleanup:
if (ret) {
for (i = 0; i < num_vnuma; i++) {
libxl_vnode_info *p = &vnuma_nodes[i];
VIR_FREE(p->distances);
}
VIR_FREE(vnuma_nodes);
}
return ret;
}
#endif
static int static int
libxlDiskSetDiscard(libxl_device_disk *x_disk, int discard) libxlDiskSetDiscard(libxl_device_disk *x_disk, int discard)
{ {
@ -2184,6 +2298,11 @@ libxlBuildDomainConfig(virPortAllocatorPtr graphicsports,
if (libxlMakeDomBuildInfo(def, ctx, caps, d_config) < 0) if (libxlMakeDomBuildInfo(def, ctx, caps, d_config) < 0)
return -1; return -1;
#ifdef LIBXL_HAVE_VNUMA
if (libxlMakeVnumaList(def, ctx, d_config) < 0)
return -1;
#endif
if (libxlMakeDiskList(def, d_config) < 0) if (libxlMakeDiskList(def, d_config) < 0)
return -1; return -1;

View File

@ -415,10 +415,15 @@ libxlDomainDefPostParse(virDomainDefPtr def,
if (xenDomainDefAddImplicitInputDevice(def) < 0) if (xenDomainDefAddImplicitInputDevice(def) < 0)
return -1; return -1;
/* For x86_64 HVM, always enable pae */ /* For x86_64 HVM */
if (def->os.type == VIR_DOMAIN_OSTYPE_HVM && if (def->os.type == VIR_DOMAIN_OSTYPE_HVM &&
def->os.arch == VIR_ARCH_X86_64) { def->os.arch == VIR_ARCH_X86_64) {
/* always enable pae */
def->features[VIR_DOMAIN_FEATURE_PAE] = VIR_TRISTATE_SWITCH_ON; def->features[VIR_DOMAIN_FEATURE_PAE] = VIR_TRISTATE_SWITCH_ON;
/* if vnuma is effective enable acpi */
if (virDomainNumaGetNodeCount(def->numa) > 0)
def->features[VIR_DOMAIN_FEATURE_ACPI] = VIR_TRISTATE_SWITCH_ON;
} }
return 0; return 0;