提交 403e8606 编写于 作者: P Peter Krempa

conf: Pre-calculate initial memory size instead of always calculating it

Add 'initial_memory' member to struct virDomainMemtune so that the
memory size can be pre-calculated once instead of inferring it always
again and again.

Separating of the fields will also allow finer granularity of decisions
in later patches where it will allow to keep the old initial memory
value in cases where we are handling incomming migration from older
versions that did not always update the size from NUMA as the code did
previously.

The change also requires modification of the qemu memory alignment
function since at the point where we are modifying the size of NUMA
nodes the total size needs to be recalculated too.

The refactoring done in this patch also fixes a crash in the hyperv
driver that did not properly initialize def->numa and thus
virDomainNumaGetMemorySize(def->numa) crashed.

In summary this patch should have no functional impact at this point.
上级 8059a990
...@@ -3728,6 +3728,15 @@ virDomainDefRemoveDuplicateMetadata(virDomainDefPtr def) ...@@ -3728,6 +3728,15 @@ virDomainDefRemoveDuplicateMetadata(virDomainDefPtr def)
static int static int
virDomainDefPostParseMemory(virDomainDefPtr def) virDomainDefPostParseMemory(virDomainDefPtr def)
{ {
size_t i;
if ((def->mem.initial_memory = virDomainNumaGetMemorySize(def->numa)) == 0) {
def->mem.initial_memory = def->mem.total_memory;
for (i = 0; i < def->nmems; i++)
def->mem.initial_memory -= def->mems[i]->size;
}
if (virDomainDefGetMemoryInitial(def) == 0) { if (virDomainDefGetMemoryInitial(def) == 0) {
virReportError(VIR_ERR_XML_ERROR, "%s", virReportError(VIR_ERR_XML_ERROR, "%s",
_("Memory size must be specified via <memory> or in the " _("Memory size must be specified via <memory> or in the "
...@@ -7699,19 +7708,7 @@ virDomainDefHasMemoryHotplug(const virDomainDef *def) ...@@ -7699,19 +7708,7 @@ virDomainDefHasMemoryHotplug(const virDomainDef *def)
unsigned long long unsigned long long
virDomainDefGetMemoryInitial(virDomainDefPtr def) virDomainDefGetMemoryInitial(virDomainDefPtr def)
{ {
unsigned long long ret; return def->mem.initial_memory;
size_t i;
/* return NUMA memory size total in case numa is enabled */
if ((ret = virDomainNumaGetMemorySize(def->numa)) > 0) {
return ret;
} else {
ret = def->mem.total_memory;
for (i = 0; i < def->nmems; i++)
ret -= def->mems[i]->size;
}
return def->mem.total_memory;
} }
...@@ -7720,13 +7717,30 @@ virDomainDefGetMemoryInitial(virDomainDefPtr def) ...@@ -7720,13 +7717,30 @@ virDomainDefGetMemoryInitial(virDomainDefPtr def)
* @def: domain definition * @def: domain definition
* @size: size to set * @size: size to set
* *
* Sets the total memory size in @def. * Sets the total memory size in @def. This function should be used only by
* hypervisors that don't support memory hotplug.
*/ */
void void
virDomainDefSetMemoryTotal(virDomainDefPtr def, virDomainDefSetMemoryTotal(virDomainDefPtr def,
unsigned long long size) unsigned long long size)
{ {
def->mem.total_memory = size; def->mem.total_memory = size;
def->mem.initial_memory = size;
}
/**
* virDomainDefSetMemoryInitial:
* @def: domain definition
* @size: size to set
*
* Sets the initial memory size (without memory modules) in @def.
*/
void
virDomainDefSetMemoryInitial(virDomainDefPtr def,
unsigned long long size)
{
def->mem.initial_memory = size;
} }
...@@ -7744,12 +7758,10 @@ virDomainDefGetMemoryActual(virDomainDefPtr def) ...@@ -7744,12 +7758,10 @@ virDomainDefGetMemoryActual(virDomainDefPtr def)
unsigned long long ret; unsigned long long ret;
size_t i; size_t i;
if ((ret = virDomainNumaGetMemorySize(def->numa)) > 0) { ret = def->mem.initial_memory;
for (i = 0; i < def->nmems; i++)
ret += def->mems[i]->size; for (i = 0; i < def->nmems; i++)
} else { ret += def->mems[i]->size;
ret = def->mem.total_memory;
}
return ret; return ret;
} }
......
...@@ -2145,6 +2145,8 @@ struct _virDomainMemtune { ...@@ -2145,6 +2145,8 @@ struct _virDomainMemtune {
/* total memory size including memory modules in kibibytes, this field /* total memory size including memory modules in kibibytes, this field
* should be accessed only via accessors */ * should be accessed only via accessors */
unsigned long long total_memory; unsigned long long total_memory;
/* initial memory size in kibibytes = total_memory excluding memory modules*/
unsigned long long initial_memory;
unsigned long long cur_balloon; /* in kibibytes, capped at ulong thanks unsigned long long cur_balloon; /* in kibibytes, capped at ulong thanks
to virDomainGetInfo */ to virDomainGetInfo */
...@@ -2324,6 +2326,7 @@ struct _virDomainDef { ...@@ -2324,6 +2326,7 @@ struct _virDomainDef {
unsigned long long virDomainDefGetMemoryInitial(virDomainDefPtr def); unsigned long long virDomainDefGetMemoryInitial(virDomainDefPtr def);
void virDomainDefSetMemoryTotal(virDomainDefPtr def, unsigned long long size); void virDomainDefSetMemoryTotal(virDomainDefPtr def, unsigned long long size);
void virDomainDefSetMemoryInitial(virDomainDefPtr def, unsigned long long size);
unsigned long long virDomainDefGetMemoryActual(virDomainDefPtr def); unsigned long long virDomainDefGetMemoryActual(virDomainDefPtr def);
bool virDomainDefHasMemoryHotplug(const virDomainDef *def); bool virDomainDefHasMemoryHotplug(const virDomainDef *def);
......
...@@ -227,6 +227,7 @@ virDomainDefParseFile; ...@@ -227,6 +227,7 @@ virDomainDefParseFile;
virDomainDefParseNode; virDomainDefParseNode;
virDomainDefParseString; virDomainDefParseString;
virDomainDefPostParse; virDomainDefPostParse;
virDomainDefSetMemoryInitial;
virDomainDefSetMemoryTotal; virDomainDefSetMemoryTotal;
virDomainDeleteConfig; virDomainDeleteConfig;
virDomainDeviceAddressIsValid; virDomainDeviceAddressIsValid;
......
...@@ -3374,6 +3374,7 @@ qemuDomainGetMemorySizeAlignment(virDomainDefPtr def ATTRIBUTE_UNUSED) ...@@ -3374,6 +3374,7 @@ qemuDomainGetMemorySizeAlignment(virDomainDefPtr def ATTRIBUTE_UNUSED)
int int
qemuDomainAlignMemorySizes(virDomainDefPtr def) qemuDomainAlignMemorySizes(virDomainDefPtr def)
{ {
unsigned long long initialmem = 0;
unsigned long long mem; unsigned long long mem;
unsigned long long align = qemuDomainGetMemorySizeAlignment(def); unsigned long long align = qemuDomainGetMemorySizeAlignment(def);
size_t ncells = virDomainNumaGetNodeCount(def->numa); size_t ncells = virDomainNumaGetNodeCount(def->numa);
...@@ -3381,13 +3382,17 @@ qemuDomainAlignMemorySizes(virDomainDefPtr def) ...@@ -3381,13 +3382,17 @@ qemuDomainAlignMemorySizes(virDomainDefPtr def)
/* align NUMA cell sizes if relevant */ /* align NUMA cell sizes if relevant */
for (i = 0; i < ncells; i++) { for (i = 0; i < ncells; i++) {
mem = virDomainNumaGetNodeMemorySize(def->numa, i); mem = VIR_ROUND_UP(virDomainNumaGetNodeMemorySize(def->numa, i), align);
virDomainNumaSetNodeMemorySize(def->numa, i, VIR_ROUND_UP(mem, align)); initialmem += mem;
virDomainNumaSetNodeMemorySize(def->numa, i, mem);
} }
/* align initial memory size */ /* align initial memory size, if NUMA is present calculate it as total of
mem = virDomainDefGetMemoryInitial(def); * individual aligned NUMA node sizes */
virDomainDefSetMemoryTotal(def, VIR_ROUND_UP(mem, align)); if (initialmem == 0)
initialmem = VIR_ROUND_UP(virDomainDefGetMemoryInitial(def), align);
virDomainDefSetMemoryInitial(def, initialmem);
def->mem.max_memory = VIR_ROUND_UP(def->mem.max_memory, align); def->mem.max_memory = VIR_ROUND_UP(def->mem.max_memory, align);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册