xenadmin/XenAdmin/Controls/Ballooning/VMMemoryControlsEdit.cs
Konstantina Chremmou 95cd300e61 CA-210190 Memory spinner: removed the label so it's easier to align
the spinner with the surrounding controls on a parent control/form.
Also, removed the icon from the spinner because in the
majority of implementations it was invisible and the extra space was just
complicating the alignment of the surrounding controls.

Signed-off-by: Konstantina Chremmou <konstantina.chremmou@citrix.com>
2018-09-26 11:06:24 +01:00

282 lines
10 KiB
C#

/* Copyright (c) Citrix Systems, Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms,
* with or without modification, are permitted provided
* that the following conditions are met:
*
* * Redistributions of source code must retain the above
* copyright notice, this list of conditions and the
* following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the
* following disclaimer in the documentation and/or other
* materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Globalization;
using System.Windows.Forms;
using XenAdmin.Core;
using XenAPI;
namespace XenAdmin.Controls.Ballooning
{
public class VMMemoryControlsEdit : UserControl
{
// The maximum value allowed for memory for this VM
private long maxMemAllowed = VM.DEFAULT_MEM_ALLOWED;
protected bool firstPaint = true;
protected List<VM> vms;
protected VM vm0;
public List<VM> VMs
{
set
{
vms = value;
if (vms == null)
return;
if (vms.Count > 0)
{
vm0 = vms[0];
SetBallooning(vm0.has_ballooning());
}
firstPaint = true;
maxMemAllowed = CalcMaxMemAllowed();
}
}
// These properties should be abstract, but you can't have an abstract class as the
// base class of a control, otherwise the control can't be edited in the Designer.
[Browsable(false)]
public virtual double dynamic_min
{
get { return 0; }
}
[Browsable(false)]
public virtual double dynamic_max
{
get { return 0; }
}
[Browsable(false)]
public virtual double static_max
{
get { return 0; }
}
protected virtual void SetBallooning(bool hasBallooning)
{
}
// A bit hacky, but needed to fix CA-35710. The ballooning dialog has to call this before
// closing, otherwise the settings on the spinners may not have taken yet.
public virtual void UnfocusSpinners()
{
SelectNextControl(Controls[0], true, true, true, true);
}
private long CalcMaxMemAllowed()
{
long ans = (vms.Count == 0 ? VM.DEFAULT_MEM_ALLOWED : vms[0].MaxMemAllowed());
foreach (VM vm in vms)
{
long max = vm.MaxMemAllowed();
if (max < ans)
ans = max;
}
return ans;
}
protected double maxDynMin = -1; // signal value for no constraint
protected void CalcMaxDynMin()
{
maxDynMin = -1;
if (vm0.power_state == vm_power_state.Running || vm0.power_state == vm_power_state.Paused)
{
// Calculate the maximum possible value of dynamic_min, by looking at all the other VMs on this host
// and obeying the constraint sum(dynamic_min) + control domain + virtualisation overhead <= total host memory
Host host = vm0.Connection.Resolve(vm0.resident_on);
if (host != null)
{
Host_metrics host_metrics = host.Connection.Resolve(host.metrics);
if (host_metrics != null)
{
double sum_dyn_min = host.memory_overhead;
foreach (VM vm in host.Connection.ResolveAll(host.resident_VMs))
{
sum_dyn_min += vm.memory_overhead;
if (vm.is_control_domain)
{
VM_metrics vmm = vm.Connection.Resolve(vm.metrics);
if (vmm != null)
sum_dyn_min += vmm.memory_actual;
}
else if (!vms.Contains(vm))
sum_dyn_min += vm.memory_dynamic_min;
}
maxDynMin = host_metrics.memory_total - sum_dyn_min;
if (maxDynMin < 0)
maxDynMin = 0;
maxDynMin /= vms.Count;
}
}
}
}
// Maximum for dynamic_min spinner: if we have set a maxDynMin, use it,
// except when constrained by vm0.memory_static_min and dynamic_max.
protected double DynMinSpinnerMax
{
get
{
double maxDM = dynamic_max;
if (maxDynMin >= 0 && maxDynMin < maxDM)
{
maxDM = maxDynMin;
if (maxDM < vm0.memory_static_min)
maxDM = vm0.memory_static_min;
}
// If we've already managed to exceed that value, don't trash the user's setting
if (maxDM < dynamic_min)
maxDM = dynamic_min;
return maxDM;
}
}
// Minimum for dynamic_min spinner: constrained by both vm0.memory_static_min, and a proportion of static_max
protected double DynMinSpinnerMin
{
get
{
double minDM = vm0.memory_static_min;
long limit = (long)(static_max * GetMemoryRatio());
if (limit > minDM)
minDM = limit;
if (minDM > dynamic_min)
minDM = dynamic_min;
return minDM;
}
}
// Static_max also has a corresponding limit: it can't go to more than 1/frac of maxDynMin,
// or there is no legal value of dynamic_min.
protected double StatMaxSpinnerMax
{
get
{
double frac = GetMemoryRatio();
double maxSM = maxDynMin >= 0 && MemorySpinnerMax * frac > (long)maxDynMin
? maxDynMin / frac
: MemorySpinnerMax;
if (maxSM < static_max)
maxSM = static_max;
return maxSM;
}
}
protected double MemorySpinnerMax
{
get
{
double server = vm0.memory_static_max;
return (server > maxMemAllowed ? server : maxMemAllowed);
}
}
public static double CalcIncrement(double static_max, string units)
{
if (units == "MB")
{
// Calculate a suitable increment if the static_max is small
int i = 1;
while (i < static_max / Util.BINARY_MEGA / 8 && i < 128)
i *= 2;
return i * Util.BINARY_MEGA;
}
else
{
return (0.1 * Util.BINARY_GIGA);
}
}
protected double GetMemoryRatio()
{
return GetMemoryRatio(vms.ToArray());
}
/// <summary>
/// Get the most constrained memory ratio for a set of VMs.
/// It is assumed they are all on the same pool.
/// </summary>
public static double GetMemoryRatio(params VM[] vms)
{
bool hvm = false;
bool pv = false;
foreach (VM vm in vms)
{
if (vm.IsHVM())
hvm = true;
else
pv = true;
}
Pool pool = Helpers.GetPoolOfOne(vms[0].Connection);
Dictionary<string, string> poolOtherConfig = Helpers.GetOtherConfig(pool);
if (!pv) // only HVM
return GetMemoryRatioValue(poolOtherConfig, true);
else if (!hvm) // only PV
return GetMemoryRatioValue(poolOtherConfig, false);
else // both types of VM: return the more constraining ratio
{
double hvmRatio = GetMemoryRatioValue(poolOtherConfig, true);
double pvRatio = GetMemoryRatioValue(poolOtherConfig, false);
return (hvmRatio >= pvRatio ? hvmRatio : pvRatio);
}
}
private static double GetMemoryRatioValue(Dictionary<string, string> poolOtherConfig, bool hvm)
{
string key = (hvm ? "memory-ratio-hvm" : "memory-ratio-pv");
string value;
double frac = -1.0;
if (poolOtherConfig.TryGetValue(key, out value))
{
try { frac = Convert.ToDouble(value, CultureInfo.InvariantCulture); }
catch (FormatException) { }
catch (OverflowException) { }
}
if (frac < 0.0 || frac > 1.0)
frac = 0.25; // default value for both HVM and PV if ratio absent or nonsensical
return frac;
}
}
}