mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2024-11-25 06:16:37 +01:00
6c584bb351
* Tidy up `SelectMultipleVMDestinationPage.cs` * CA-375532 & CA-336510: Add warning when importing VMs with too many vCPUs * CA-375532 & CA-336510: Prevent users from starting appliance if a VM has too many vCPUs * Tidy up `Page_CpuMem`: use expression-bodied properties * Tidy up `Page_CpuMem`: use conventional naming scheme * Tidy up `Page_CpuMem`: use explicit modifiers * Tidy up `Page_CpuMem`: use `en-US` locale for names * Tidy up `Page_CpuMem`: use `var` when possible * Tidy up `Page_CpuMem`: rename `Vcpus` to `VCpus` * Tidy up `Page_CpuMem`: reorder elements * Tidy up `Page_CpuMem`: revert class renaming and fix `vCpus` typos * CA-375532: Prevent users from starting VM if they select too many vCPUs * Tidy up `Page_Finish`: use expression-bodied properties * Tidy up `Page_Finish`: fix typo * Tidy up `Page_CpuMem`: rename `CanStartVM` to `CanStartVm` * Remove unnecessary using directives * Anchor warning icon to the top-left corner of its parent * CA-375532 & CA-336510: Warn users when selecting more vCPUs than pCPUs in the Properties page * CP-41825: Add warning for imported VMs with > 32 vCPUs * CP-41825: Add warning for new VMs with > 32 vCPUs * CP-41825: Add warning for existing VMs when selecting > 32 vCPUs * CP-41825: Update wording used to alert users running VMs with > 32vCPUs * Remove unused local variable in `VappStartCommand.cs` * Reword some vCPUs messages * Simplify assignment of `CanStartImmediately` in `Page_Finish.cs` * Simplify assignment of `CanStartVmsAutomatically` in `ImportFinishPage.cs` * Remove useless `Count` check in `VappStartCommand` * Rename `pictureBox1` to `warningPictureBox` and change its `SizeMode` * Separate CPU and memory warnings in `Page_CpuMem` * Shorten warning messages shown in `Page_CpuMem` * Disable option to start VM when memory configuration isn't valid * Report number of vCPUs and pCPUs when showing warning in New VM wizard * Tidy up `CpuMemoryEditPage`: rename elements * Tidy up `CpuMemoryEditPage`: use explicity modifiers * Tidy up `CpuMemoryEditPage`: Rename `CPUMemoryEditPage` to `CpuMemoryEditPage` * Tidy up `CpuMemoryEditPage`: Miscellaneous renames and improvements * Tidy up `CpuMemoryEditPage`: Fix whitespace * Tidy up `CpuMemoryEditPage`: Use ReSharper code formatter * Tidy up `CpuMemoryEditPage`: Move events to own region * Tidy up `CpuMemoryEditPage`: Move `IEditPage` members to own region * Tidy up `CpuMemoryEditPage`: Order members outside of regions * Replace topology and vCPU warning labels with structured warnings * Ensure warnings ends with a dot * Allow multiple CPU warnings to show at the same time * Remove warning link for vCPUS > pCPUS Option is not achievable * Enable option to force unit used in `MemorySpinner` * Replace `NumericUpDown` with `MemorySpinner` and show warnings at bottom of control * Check destination pCPUs count when destination is not a pool * Do not disable option to start VMs after import if user selects a shared SR * Add memory warning when importing appliance with too much memory * Sort vCPUs and memory warning strings * Remove target from `ImportWizard`'s summary Target is not useful and can cause confusion as appliance VMs could start on non-target hosts * The home server is needed in the summary when importing XVA. The target server is not needed in the summary when migrating a VM. * Update warnings when the server selection changes. Corrected memory calculation. * Show all warnings for vCPUs. Moved topology warning to the bottom. Separated methods for showing memory and vCPU warnings. * Removed the memory setting from the VM Properties dialog. Split running VM info to a separate panel. * Corrections to the logic for starting the VM automatically. Also, Fixed a bit the layout of the Finish page to make better use of the available space. * Corrections as per code review. Signed-off-by: Danilo Del Busso <danilo.delbusso@cloud.com> Co-authored-by: Konstantina Chremmou <Konstantina.Chremmou@cloud.com>
509 lines
21 KiB
C#
509 lines
21 KiB
C#
/* Copyright (c) Cloud Software Group, Inc.
|
|
*
|
|
* 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.Text;
|
|
using System.Windows.Forms;
|
|
using XenAdmin.Controls;
|
|
using XenAdmin.Controls.Ballooning;
|
|
using XenAdmin.Core;
|
|
using XenAPI;
|
|
|
|
namespace XenAdmin.Wizards.NewVMWizard
|
|
{
|
|
public partial class Page_CpuMem : XenTabPage
|
|
{
|
|
private VM _template;
|
|
|
|
// number of spinners to show
|
|
private enum MemoryMode
|
|
{
|
|
JustMemory = 1,
|
|
MinimumAndMaximum = 2,
|
|
MinimumMaximumAndStaticMax = 3
|
|
}
|
|
|
|
private MemoryMode _memoryMode = MemoryMode.JustMemory;
|
|
|
|
private double _memoryRatio; // the permitted ratio of dynamic_min / static_max
|
|
private bool _initializing = true;
|
|
private bool _isVCpuHotplugSupported;
|
|
private int _minVCpus;
|
|
private long _maxVCpus;
|
|
private long _maxMemTotal;
|
|
private long _maxMemFree;
|
|
private long _prevVCpusMax;
|
|
public VM SelectedTemplate { private get; set; }
|
|
|
|
// Please note that the comboBoxVCPUs control can represent two different VM properties, depending whether the VM supports vCPU hotplug or not:
|
|
// When vCPU hotplug is not supported, comboBoxVCPUs represents the initial number of vCPUs (VCPUs_at_startup). In this case we will also set the VM property VCPUs_max to the same value.
|
|
// When vCPU hotplug is supported, comboBoxVCPUs represents the maximum number of vCPUs (VCPUs_max). And the initial number of vCPUs is represented in comboBoxInitialVCPUs (which is only visible in this case)
|
|
|
|
public Page_CpuMem()
|
|
{
|
|
InitializeComponent();
|
|
}
|
|
|
|
public override string Text => Messages.NEWVMWIZARD_CPUMEMPAGE_NAME;
|
|
|
|
public override string PageTitle => Messages.NEWVMWIZARD_CPUMEMPAGE_TITLE;
|
|
|
|
public override string HelpID => "CPU&Memory";
|
|
|
|
public double SelectedMemoryDynamicMin => spinnerDynMin.Value;
|
|
|
|
public double SelectedMemoryDynamicMax => _memoryMode == MemoryMode.JustMemory ? spinnerDynMin.Value : spinnerDynMax.Value;
|
|
|
|
public bool CanStartVm => _maxVCpus > 0 && SelectedVCpusMax <= _maxVCpus && _maxMemFree > 0 && SelectedMemoryDynamicMin <= _maxMemFree && _maxMemTotal > 0 && SelectedMemoryDynamicMin <= _maxMemTotal;
|
|
|
|
public double SelectedMemoryStaticMax =>
|
|
_memoryMode == MemoryMode.JustMemory ? spinnerDynMin.Value :
|
|
_memoryMode == MemoryMode.MinimumAndMaximum ? spinnerDynMax.Value :
|
|
spinnerStatMax.Value;
|
|
|
|
public long SelectedVCpusMax => comboBoxVCPUs.SelectedItem == null ? -1 : (long)comboBoxVCPUs.SelectedItem;
|
|
|
|
public long SelectedVCpusAtStartup
|
|
{
|
|
get
|
|
{
|
|
if (_isVCpuHotplugSupported)
|
|
return comboBoxInitialVCPUs.SelectedItem == null ? -1 : (long)comboBoxInitialVCPUs.SelectedItem;
|
|
|
|
return comboBoxVCPUs.SelectedItem == null ? -1 : (long)comboBoxVCPUs.SelectedItem;
|
|
}
|
|
}
|
|
|
|
public long SelectedCoresPerSocket => comboBoxTopology.CoresPerSocket;
|
|
|
|
protected override void PageLoadedCore(PageLoadedDirection direction)
|
|
{
|
|
if (SelectedTemplate == _template)
|
|
return;
|
|
|
|
_initializing = true;
|
|
|
|
_template = SelectedTemplate;
|
|
if (_template.SupportsBallooning() && !Helpers.FeatureForbidden(_template, Host.RestrictDMC))
|
|
_memoryMode = _template.memory_dynamic_max == _template.memory_static_max ? MemoryMode.MinimumAndMaximum : MemoryMode.MinimumMaximumAndStaticMax;
|
|
else
|
|
_memoryMode = MemoryMode.JustMemory;
|
|
|
|
_memoryRatio = VMMemoryControlsEdit.GetMemoryRatio(_template);
|
|
|
|
FreeSpinnerLimits();
|
|
|
|
if (_memoryMode == MemoryMode.JustMemory)
|
|
{
|
|
spinnerDynMin.Initialize(_template.memory_static_max, _template.memory_static_max);
|
|
labelDynMin.Text = Messages.MEMORY_COLON;
|
|
}
|
|
else
|
|
{
|
|
labelDynMin.Text = Messages.DYNAMIC_MIN_COLON;
|
|
spinnerDynMin.Initialize(_template.memory_dynamic_min, _template.memory_static_max);
|
|
FreeSpinnerLimits(); // same as CA-33831
|
|
spinnerDynMax.Initialize(_template.memory_dynamic_max, _template.memory_static_max);
|
|
if (_memoryMode == MemoryMode.MinimumMaximumAndStaticMax)
|
|
{
|
|
FreeSpinnerLimits();
|
|
spinnerStatMax.Initialize(_template.memory_static_max, _template.memory_static_max);
|
|
}
|
|
}
|
|
labelDynMaxInfo.Visible = labelDynMax.Visible = spinnerDynMax.Visible = _memoryMode == MemoryMode.MinimumAndMaximum || _memoryMode == MemoryMode.MinimumMaximumAndStaticMax;
|
|
labelStatMaxInfo.Visible = labelStatMax.Visible = spinnerStatMax.Visible = _memoryMode == MemoryMode.MinimumMaximumAndStaticMax;
|
|
|
|
_isVCpuHotplugSupported = _template.SupportsVcpuHotplug();
|
|
_minVCpus = _template.MinVCPUs();
|
|
|
|
_prevVCpusMax = _template.VCPUs_max; // we use variable in RefreshCurrentVCPUs for checking if VcpusAtStartup and VcpusMax were equal before VcpusMax changed
|
|
|
|
label5.Text = GetRubric();
|
|
|
|
InitialiseVCpuControls();
|
|
SetSpinnerLimitsAndIncrement();
|
|
ValidateMemorySettings();
|
|
ValidateVCpuSettings();
|
|
ValidateInitialVCpuSettings();
|
|
OnPageUpdated();
|
|
|
|
_initializing = false;
|
|
}
|
|
|
|
public override void SelectDefaultControl()
|
|
{
|
|
comboBoxVCPUs.Select();
|
|
}
|
|
|
|
private void InitialiseVCpuControls()
|
|
{
|
|
labelVCPUs.Text = _isVCpuHotplugSupported
|
|
? Messages.VM_CPUMEMPAGE_MAX_VCPUS_LABEL
|
|
: Messages.VM_CPUMEMPAGE_VCPUS_LABEL;
|
|
|
|
labelInitialVCPUs.Text = Messages.VM_CPUMEMPAGE_INITIAL_VCPUS_LABEL;
|
|
labelInitialVCPUs.Visible = comboBoxInitialVCPUs.Visible = _isVCpuHotplugSupported;
|
|
|
|
comboBoxTopology.Populate(_template.VCPUs_at_startup, _template.VCPUs_max, _template.GetCoresPerSocket(), _template.MaxCoresPerSocket());
|
|
PopulateVCpus(_template.MaxVCPUsAllowed(), _isVCpuHotplugSupported ? _template.VCPUs_max : _template.VCPUs_at_startup);
|
|
|
|
if (_isVCpuHotplugSupported)
|
|
PopulateVCpusAtStartup(_template.VCPUs_max, _template.VCPUs_at_startup);
|
|
}
|
|
|
|
private void PopulateVCpuComboBox(ComboBox comboBox, long min, long max, long currentValue, Predicate<long> isValid)
|
|
{
|
|
comboBox.BeginUpdate();
|
|
comboBox.Items.Clear();
|
|
for (var i = min; i <= max; ++i)
|
|
{
|
|
if (i == currentValue || isValid(i))
|
|
comboBox.Items.Add(i);
|
|
}
|
|
if (currentValue > max)
|
|
comboBox.Items.Add(currentValue);
|
|
comboBox.SelectedItem = currentValue;
|
|
comboBox.EndUpdate();
|
|
}
|
|
|
|
private void PopulateVCpus(long maxVCpus, long currentVCpus)
|
|
{
|
|
PopulateVCpuComboBox(comboBoxVCPUs, 1, maxVCpus, currentVCpus, i => comboBoxTopology.IsValidVCPU(i));
|
|
}
|
|
|
|
private void PopulateVCpusAtStartup(long max, long currentValue)
|
|
{
|
|
PopulateVCpuComboBox(comboBoxInitialVCPUs, 1, max, currentValue, i => true);
|
|
}
|
|
|
|
private string GetRubric()
|
|
{
|
|
var sb = new StringBuilder();
|
|
sb.Append(Messages.NEWVMWIZARD_CPUMEMPAGE_RUBRIC);
|
|
// add hotplug text
|
|
if (_isVCpuHotplugSupported)
|
|
sb.Append(Messages.VM_CPUMEMPAGE_RUBRIC_HOTPLUG);
|
|
return sb.ToString();
|
|
}
|
|
|
|
// Return the larger of the template's MaxMemAllowed and the template's static max,
|
|
// to avoid crashes in the spinners (CA-40041).
|
|
private long MaxMemAllowed
|
|
{
|
|
get
|
|
{
|
|
var msm = _template.memory_static_max;
|
|
var mma = _template.MaxMemAllowed();
|
|
return (msm > mma ? msm : mma);
|
|
}
|
|
}
|
|
|
|
private void FreeSpinnerLimits()
|
|
{
|
|
var maxMemAllowed = MaxMemAllowed;
|
|
spinnerDynMin.SetRange(0, maxMemAllowed);
|
|
spinnerDynMax.SetRange(0, maxMemAllowed);
|
|
spinnerStatMax.SetRange(0, maxMemAllowed);
|
|
}
|
|
|
|
private void SetSpinnerLimitsAndIncrement()
|
|
{
|
|
spinnerDynMin.Increment = spinnerDynMax.Increment = spinnerStatMax.Increment = VMMemoryControlsEdit.CalcIncrement(_template.memory_static_max, spinnerDynMin.Units);
|
|
|
|
var maxMemAllowed = MaxMemAllowed;
|
|
double min = _template.memory_static_min;
|
|
if (_memoryMode == MemoryMode.JustMemory)
|
|
{
|
|
spinnerDynMin.SetRange(min, maxMemAllowed);
|
|
ShowMemoryMinMaxInformation(labelDynMinInfo, min, maxMemAllowed);
|
|
return;
|
|
}
|
|
var min2 = (long)(SelectedMemoryStaticMax * _memoryRatio);
|
|
if (min < min2)
|
|
min = min2;
|
|
var max = SelectedMemoryDynamicMax;
|
|
if (max < min)
|
|
max = min;
|
|
spinnerDynMin.SetRange(min, max);
|
|
ShowMemoryMinMaxInformation(labelDynMinInfo, min, max);
|
|
|
|
spinnerDynMax.SetRange(SelectedMemoryDynamicMin,
|
|
_memoryMode == MemoryMode.MinimumAndMaximum ? maxMemAllowed : SelectedMemoryStaticMax);
|
|
ShowMemoryMinMaxInformation(labelDynMaxInfo, SelectedMemoryDynamicMin,
|
|
_memoryMode == MemoryMode.MinimumAndMaximum ? maxMemAllowed : SelectedMemoryStaticMax);
|
|
|
|
spinnerStatMax.SetRange(SelectedMemoryDynamicMax, maxMemAllowed);
|
|
ShowMemoryMinMaxInformation(labelStatMaxInfo, SelectedMemoryDynamicMax, maxMemAllowed);
|
|
}
|
|
|
|
public void DisableMemoryControls()
|
|
{
|
|
spinnerDynMin.Enabled = false;
|
|
spinnerDynMax.Enabled = false;
|
|
spinnerStatMax.Enabled = false;
|
|
}
|
|
|
|
public override List<KeyValuePair<string, string>> PageSummary
|
|
{
|
|
get
|
|
{
|
|
var sum = new List<KeyValuePair<string, string>>();
|
|
|
|
if (_isVCpuHotplugSupported)
|
|
{
|
|
sum.Add(new KeyValuePair<string, string>(Messages.NEWVMWIZARD_CPUMEMPAGE_MAX_VCPUS, SelectedVCpusMax.ToString()));
|
|
sum.Add(new KeyValuePair<string, string>(Messages.NEWVMWIZARD_CPUMEMPAGE_INITIAL_VCPUS, SelectedVCpusAtStartup.ToString()));
|
|
}
|
|
else
|
|
{
|
|
sum.Add(new KeyValuePair<string, string>(Messages.NEWVMWIZARD_CPUMEMPAGE_VCPUS, SelectedVCpusAtStartup.ToString()));
|
|
}
|
|
sum.Add(new KeyValuePair<string, string>(Messages.NEWVMWIZARD_CPUMEMPAGE_TOPOLOGY, comboBoxTopology.Text));
|
|
if (_memoryMode == MemoryMode.JustMemory)
|
|
sum.Add(new KeyValuePair<string, string>(Messages.MEMORY, Util.MemorySizeStringSuitableUnits(SelectedMemoryStaticMax, false)));
|
|
else
|
|
{
|
|
sum.Add(new KeyValuePair<string, string>(Messages.DYNAMIC_MIN, Util.MemorySizeStringSuitableUnits(SelectedMemoryDynamicMin, false)));
|
|
sum.Add(new KeyValuePair<string, string>(Messages.DYNAMIC_MAX, Util.MemorySizeStringSuitableUnits(SelectedMemoryDynamicMax, false)));
|
|
if (_memoryMode == MemoryMode.MinimumMaximumAndStaticMax)
|
|
sum.Add(new KeyValuePair<string, string>(Messages.STATIC_MAX, Util.MemorySizeStringSuitableUnits(SelectedMemoryStaticMax, false)));
|
|
}
|
|
return sum;
|
|
}
|
|
}
|
|
|
|
private void ValidateMemorySettings()
|
|
{
|
|
Host maxMemTotalHost = null;
|
|
Host maxMemFreeHost = null;
|
|
_maxMemTotal = 0;
|
|
_maxMemFree = 0;
|
|
|
|
foreach (var host in Connection.Cache.Hosts)
|
|
{
|
|
var metrics = Connection.Resolve(host.metrics);
|
|
|
|
if (metrics != null && metrics.memory_total > _maxMemTotal)
|
|
{
|
|
_maxMemTotal = metrics.memory_total;
|
|
maxMemTotalHost = host;
|
|
}
|
|
|
|
// The available memory of a server is taken to be the current memory_free,
|
|
// plus however much we can squeeze down the existing VMs. This assumes
|
|
// that the overhead won't increase when we create the new VM, so it
|
|
// has false negatives.
|
|
var memoryFree = host.memory_available_calc();
|
|
|
|
if (metrics != null && memoryFree > _maxMemFree)
|
|
{
|
|
_maxMemFree = memoryFree;
|
|
maxMemFreeHost = host;
|
|
}
|
|
}
|
|
|
|
if (maxMemTotalHost != null && SelectedMemoryDynamicMin > _maxMemTotal)
|
|
{
|
|
ShowMemoryWarning(string.Format(Messages.NEWVMWIZARD_CPUMEMPAGE_MEMORYWARN_TOTAL, Util.MemorySizeStringSuitableUnits(_maxMemTotal, false)));
|
|
}
|
|
else if (maxMemFreeHost != null && SelectedMemoryDynamicMin > _maxMemFree)
|
|
{
|
|
ShowMemoryWarning(string.Format(Messages.NEWVMWIZARD_CPUMEMPAGE_MEMORYWARN_FREE, Util.MemorySizeStringSuitableUnits(_maxMemFree, false)));
|
|
}
|
|
else
|
|
{
|
|
ShowMemoryWarning();
|
|
}
|
|
}
|
|
|
|
private void ValidateVCpuSettings()
|
|
{
|
|
Host maxVCpusHost = null;
|
|
_maxVCpus = 0;
|
|
|
|
var warnings = new List<string>();
|
|
|
|
foreach (var host in Connection.Cache.Hosts)
|
|
{
|
|
long hostCpus = 0;
|
|
|
|
foreach (var cpu in Connection.Cache.Host_cpus)
|
|
{
|
|
if (cpu.host.opaque_ref.Equals(host.opaque_ref))
|
|
hostCpus++;
|
|
}
|
|
|
|
if (hostCpus > _maxVCpus)
|
|
{
|
|
_maxVCpus = hostCpus;
|
|
maxVCpusHost = host;
|
|
}
|
|
}
|
|
|
|
if (maxVCpusHost != null && SelectedVCpusMax > _maxVCpus)
|
|
{
|
|
var isStandAloneHost = Helpers.GetPool(maxVCpusHost.Connection) == null;
|
|
if (isStandAloneHost)
|
|
{
|
|
warnings.Add(string.Format(Messages.NEWVMWIZARD_CPUMEMPAGE_VCPUSWARN_STANDALONE_HOST, SelectedVCpusMax, _maxVCpus));
|
|
}
|
|
else
|
|
{
|
|
warnings.Add(string.Format(Messages.NEWVMWIZARD_CPUMEMPAGE_VCPUSWARN_POOL, SelectedVCpusMax, _maxVCpus));
|
|
}
|
|
}
|
|
|
|
if (SelectedVCpusMax > VM.MAX_VCPUS_FOR_NON_TRUSTED_VMS)
|
|
{
|
|
warnings.Add(string.Format(Messages.VCPUS_UNTRUSTED_VM_WARNING, VM.MAX_VCPUS_FOR_NON_TRUSTED_VMS, BrandManager.BrandConsole));
|
|
}
|
|
|
|
ShowCpuWarning(string.Join("\n\n", warnings));
|
|
|
|
if (SelectedVCpusMax < _minVCpus)
|
|
{
|
|
vCPUWarningLabel.Text = string.Format(Messages.VM_CPUMEMPAGE_VCPU_MIN_WARNING, _minVCpus);
|
|
vCPUWarningLabel.Visible = true;
|
|
}
|
|
else
|
|
{
|
|
vCPUWarningLabel.Visible = false;
|
|
}
|
|
}
|
|
|
|
private void ValidateInitialVCpuSettings()
|
|
{
|
|
if (SelectedVCpusAtStartup < _minVCpus)
|
|
{
|
|
initialVCPUWarningLabel.Text = string.Format(Messages.VM_CPUMEMPAGE_VCPU_MIN_WARNING, _minVCpus);
|
|
initialVCPUWarningLabel.Visible = true;
|
|
}
|
|
else
|
|
{
|
|
initialVCPUWarningLabel.Visible = false;
|
|
}
|
|
}
|
|
|
|
private void ValidateTopologySettings()
|
|
{
|
|
if (comboBoxVCPUs.SelectedItem != null)
|
|
ShowTopologyWarning(VM.ValidVCPUConfiguration((long)comboBoxVCPUs.SelectedItem, comboBoxTopology.CoresPerSocket));
|
|
}
|
|
|
|
private void ShowMemoryWarning(string text = null)
|
|
{
|
|
var show = !string.IsNullOrEmpty(text);
|
|
memoryWarningLabel.Text = show ? text : null;
|
|
memoryPictureBox.Visible = memoryWarningLabel.Visible = show;
|
|
}
|
|
|
|
private void ShowTopologyWarning(string text = null)
|
|
{
|
|
var show = !string.IsNullOrEmpty(text);
|
|
labelTopologyWarning.Text = show ? text : null;
|
|
pictureBoxTopology.Visible = labelTopologyWarning.Visible = show;
|
|
}
|
|
|
|
private void ShowCpuWarning(string text = null)
|
|
{
|
|
var show = !string.IsNullOrEmpty(text);
|
|
cpuWarningLabel.Text = show ? text : null;
|
|
cpuWarningPictureBox.Visible = cpuWarningLabel.Visible = show;
|
|
}
|
|
|
|
private void ShowMemoryMinMaxInformation(Label label, double min, double max)
|
|
{
|
|
label.Text = string.Format(
|
|
Messages.NEWVMWIZARD_CPUMEMPAGE_MEMORYINFO,
|
|
FormatMemory(min),
|
|
FormatMemory(max));
|
|
}
|
|
|
|
private string FormatMemory(double numberOfBytes)
|
|
{
|
|
return Util.MemorySizeStringSuitableUnits(numberOfBytes, true);
|
|
}
|
|
|
|
#region Control event handlers
|
|
|
|
private void vCPU_ValueChanged(object sender, EventArgs e)
|
|
{
|
|
comboBoxTopology.Update((long)comboBoxVCPUs.SelectedItem);
|
|
ValidateVCpuSettings();
|
|
RefreshCurrentVCpus();
|
|
OnPageUpdated();
|
|
}
|
|
|
|
private void memory_ValueChanged(object sender, EventArgs e)
|
|
{
|
|
if (_initializing)
|
|
return;
|
|
|
|
SetSpinnerLimitsAndIncrement();
|
|
ValidateMemorySettings();
|
|
OnPageUpdated();
|
|
}
|
|
|
|
private void RefreshCurrentVCpus()
|
|
{
|
|
// refresh comboBoxInitialVCPUs if it's visible and populated
|
|
if (comboBoxInitialVCPUs.Visible && comboBoxInitialVCPUs.Items.Count > 0)
|
|
{
|
|
// VcpusAtStartup is always <= VcpusMax
|
|
// So if VcpusMax is decreased below VcpusAtStartup, then VcpusAtStartup is decreased to that number too
|
|
// If VcpusAtStartup and VcpusMax are equal, and VcpusMax is changed, then VcpusAtStartup is changed to match
|
|
// But if the numbers are unequal, and VcpusMax is changed but is still higher than VcpusAtStartup, then VcpusAtStartup is unchanged
|
|
var newValue = SelectedVCpusAtStartup;
|
|
|
|
if (SelectedVCpusMax < SelectedVCpusAtStartup)
|
|
newValue = SelectedVCpusMax;
|
|
else if (SelectedVCpusAtStartup == _prevVCpusMax && SelectedVCpusMax != _prevVCpusMax)
|
|
newValue = SelectedVCpusMax;
|
|
|
|
PopulateVCpusAtStartup(SelectedVCpusMax, newValue);
|
|
_prevVCpusMax = SelectedVCpusMax;
|
|
}
|
|
}
|
|
|
|
private void comboBoxTopology_SelectedIndexChanged(object sender, EventArgs e)
|
|
{
|
|
ValidateTopologySettings();
|
|
}
|
|
|
|
private void comboBoxInitialVCPUs_SelectedIndexChanged(object sender, EventArgs e)
|
|
{
|
|
ValidateInitialVCpuSettings();
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|