mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-25 14:05:18 +00:00
Handle non-sequential NUMA node numbers
On some platforms like IBM PowerNV the NUMA node numbers can be non-sequential. For eg. numactl --hardware o/p from such a machine looks as given below node distances: node 0 1 16 17 0: 10 40 40 40 1: 40 10 40 40 16: 40 40 10 40 17: 40 40 40 10 The NUMA nodes are 0,1,16,17 Libvirt uses sequential index as NUMA node numbers and this can result in crash or incorrect results. Signed-off-by: Shivaprasad G Bhat <sbhat@linux.vnet.ibm.com> Signed-off-by: Pradipta Kr. Banerjee <bpradip@in.ibm.com>
This commit is contained in:
parent
037ffda3c7
commit
cd921cf077
@ -1000,10 +1000,18 @@ virCapabilitiesGetCpusForNode(virCapsPtr caps,
|
|||||||
size_t node,
|
size_t node,
|
||||||
virBitmapPtr cpumask)
|
virBitmapPtr cpumask)
|
||||||
{
|
{
|
||||||
virCapsHostNUMACellPtr cell = caps->host.numaCell[node];
|
virCapsHostNUMACellPtr cell = NULL;
|
||||||
size_t cpu;
|
size_t cpu;
|
||||||
|
size_t index;
|
||||||
|
/* The numa node numbers can be non-contiguous. Ex: 0,1,16,17. */
|
||||||
|
for (index = 0; index < caps->host.nnumaCell; index++) {
|
||||||
|
if (caps->host.numaCell[index]->num == node) {
|
||||||
|
cell = caps->host.numaCell[index];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (cpu = 0; cpu < cell->ncpus; cpu++) {
|
for (cpu = 0; cell && cpu < cell->ncpus; cpu++) {
|
||||||
if (virBitmapSetBit(cpumask, cell->cpus[cpu].id) < 0) {
|
if (virBitmapSetBit(cpumask, cell->cpus[cpu].id) < 0) {
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
||||||
_("Cpu '%u' in node '%zu' is out of range "
|
_("Cpu '%u' in node '%zu' is out of range "
|
||||||
|
@ -8566,12 +8566,13 @@ qemuDomainSetNumaParamsLive(virDomainObjPtr vm,
|
|||||||
|
|
||||||
for (i = 0; i < caps->host.nnumaCell; i++) {
|
for (i = 0; i < caps->host.nnumaCell; i++) {
|
||||||
bool result;
|
bool result;
|
||||||
if (virBitmapGetBit(nodeset, i, &result) < 0) {
|
virCapsHostNUMACellPtr cell = caps->host.numaCell[i];
|
||||||
|
if (virBitmapGetBit(nodeset, cell->num, &result) < 0) {
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
||||||
_("Failed to get cpuset bit values"));
|
_("Failed to get cpuset bit values"));
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
if (result && (virBitmapSetBit(temp_nodeset, i) < 0)) {
|
if (result && (virBitmapSetBit(temp_nodeset, cell->num) < 0)) {
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
||||||
_("Failed to set temporary cpuset bit values"));
|
_("Failed to set temporary cpuset bit values"));
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
@ -2002,7 +2002,7 @@ qemuPrepareCpumap(virQEMUDriverPtr driver,
|
|||||||
size_t j;
|
size_t j;
|
||||||
int cur_ncpus = caps->host.numaCell[i]->ncpus;
|
int cur_ncpus = caps->host.numaCell[i]->ncpus;
|
||||||
bool result;
|
bool result;
|
||||||
if (virBitmapGetBit(nodemask, i, &result) < 0) {
|
if (virBitmapGetBit(nodemask, caps->host.numaCell[i]->num, &result) < 0) {
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
||||||
_("Failed to convert nodeset to cpuset"));
|
_("Failed to convert nodeset to cpuset"));
|
||||||
virBitmapFree(cpumap);
|
virBitmapFree(cpumap);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user