mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2024-12-22 00:16:03 +01:00
92f0499911
We observed that some threads can reach deadlock-ish state after they have Invoked into a control's UI thread. When it happens they are all in a waiting for join or in sleep state for very long time, although there should not be any deadlock situations.
It seems this has something to do with multiple parent controls and with which control we invoked on. This should not make a difference, because we have got one UI thread (for MainWindow) they should wait for, but we have seen it does.
The solution that fixed this issue was to invoke on the MainWindow instead of various controls (see a4fe507adf
).
This changeset is changing all our Invokes to invoke into MainWindow
instead of a control itself. (MainWindow's UI thread is the only UI thread
all Control is using in XenCenter)
This changeset should be in place until we have found the root cause or the exact reason for the above.
325 lines
9.8 KiB
C#
325 lines
9.8 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.ComponentModel;
|
|
using System.Threading;
|
|
using System.Windows.Forms;
|
|
using XenAdmin.Actions;
|
|
using XenAdmin.Controls;
|
|
using XenAdmin.Core;
|
|
using XenAdmin.Dialogs;
|
|
using XenAdmin.Network;
|
|
using XenAPI;
|
|
|
|
|
|
namespace XenAdmin.Wizards.ImportWizard
|
|
{
|
|
public partial class StoragePickerPage : XenTabPage
|
|
{
|
|
#region Private fields
|
|
private Host m_targetHost;
|
|
private IXenConnection m_targetConnection;
|
|
private volatile Task m_importTask;
|
|
private bool m_alreadyFoundVM;
|
|
private ActionProgressDialog m_actionDialog;
|
|
private string m_buttonNextText = Messages.WIZARD_BUTTON_NEXT;
|
|
private bool m_buttonPreviousEnabled = true;
|
|
private bool m_buttonNextEnabled;
|
|
#endregion
|
|
|
|
public event Action ImportVmCompleted;
|
|
|
|
public StoragePickerPage()
|
|
{
|
|
InitializeComponent();
|
|
}
|
|
|
|
#region Base class (XenTabPage) overrides
|
|
|
|
/// <summary>
|
|
/// Gets the page's title (headline)
|
|
/// </summary>
|
|
public override string PageTitle { get { return Messages.IMPORT_SELECT_STORAGE_PAGE_TITLE; } }
|
|
|
|
/// <summary>
|
|
/// Gets the page's label in the (left hand side) wizard progress panel
|
|
/// </summary>
|
|
public override string Text { get { return Messages.IMPORT_SELECT_STORAGE_PAGE_TEXT; } }
|
|
|
|
/// <summary>
|
|
/// Gets the value by which the help files section for this page is identified
|
|
/// </summary>
|
|
public override string HelpID { get { return "StoragePicker"; } }
|
|
|
|
protected override bool ImplementsIsDirty()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
public override void PageLoaded(PageLoadedDirection direction)
|
|
{
|
|
base.PageLoaded(direction);
|
|
m_buttonNextText = (direction == PageLoadedDirection.Forward) ? Messages.IMPORT_VM_IMPORT : Messages.WIZARD_BUTTON_NEXT;
|
|
}
|
|
|
|
public override void PageLeave(PageLoadedDirection direction, ref bool cancel)
|
|
{
|
|
if (direction == PageLoadedDirection.Forward && IsDirty)
|
|
{
|
|
cancel = true;
|
|
IsDirty = false;
|
|
SetButtonNextEnabled(false);
|
|
m_buttonPreviousEnabled = false;
|
|
OnPageUpdated();
|
|
ImportXvaAction = new ImportVmAction(m_targetHost == null ? m_targetConnection : m_targetHost.Connection,
|
|
m_targetHost,
|
|
FilePath,
|
|
SR);
|
|
ImportXvaAction.Completed += m_importXvaAction_Completed;
|
|
|
|
m_actionDialog = new ActionProgressDialog(ImportXvaAction, ProgressBarStyle.Blocks) {ShowCancel = true};
|
|
// Now we want to start a scanning thread to search for the task ref on the vms
|
|
ThreadPool.QueueUserWorkItem(ScanTask, null);
|
|
m_alreadyFoundVM = false;
|
|
m_actionDialog.Show(this);
|
|
}
|
|
base.PageLeave(direction, ref cancel);
|
|
}
|
|
|
|
public override string NextText(bool isLastPage)
|
|
{
|
|
return m_buttonNextText;
|
|
}
|
|
|
|
public override void PopulatePage()
|
|
{
|
|
m_srPicker.SrHint.Text = Messages.IMPORT_VM_SELECT_SR;
|
|
|
|
// Select default SR
|
|
Pool pool = Helpers.GetPoolOfOne(m_targetConnection);
|
|
if (pool != null)
|
|
m_srPicker.DefaultSR = m_targetConnection.Resolve(pool.default_SR);
|
|
|
|
m_srPicker.selectDefaultSROrAny();
|
|
IsDirty = true;
|
|
}
|
|
|
|
public override bool EnablePrevious()
|
|
{
|
|
return m_buttonPreviousEnabled;
|
|
}
|
|
|
|
public override bool EnableNext()
|
|
{
|
|
return m_buttonNextEnabled;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Accessors
|
|
|
|
public ImportVmAction ImportXvaAction { get; private set; }
|
|
|
|
public string FilePath { private get; set; }
|
|
|
|
public SR SR { get { return m_srPicker.SR; } }
|
|
|
|
public VM ImportedVm { get; private set; }
|
|
|
|
#endregion
|
|
|
|
/// <summary>
|
|
/// Should be called before the Affinity is set.
|
|
/// </summary>
|
|
public void SetConnection(IXenConnection con)
|
|
{
|
|
m_targetConnection = con;
|
|
m_srPicker.Connection = con;
|
|
}
|
|
|
|
public void SetTargetHost(Host host)
|
|
{
|
|
m_targetHost = host;
|
|
m_srPicker.SetAffinity(host);
|
|
}
|
|
|
|
#region Private methods
|
|
|
|
private void SetButtonNextEnabled(bool enabled)
|
|
{
|
|
m_buttonNextEnabled = enabled;
|
|
OnPageUpdated();
|
|
}
|
|
|
|
private bool ImportInProgress()
|
|
{
|
|
// Check if the upload action has started
|
|
if (ImportXvaAction != null)
|
|
{
|
|
// if the VM has been uploaded we can continue
|
|
if (m_alreadyFoundVM)
|
|
return false;
|
|
|
|
// if it has failed or been cancelled we let them try again
|
|
if (ImportXvaAction.IsCompleted)
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
private void ScanTask(object obj)
|
|
{
|
|
Program.AssertOffEventThread();
|
|
|
|
if (ImportXvaAction == null)
|
|
return;
|
|
|
|
while (ImportXvaAction.RelatedTask == null)
|
|
Thread.Sleep(100);
|
|
|
|
while ((m_importTask = m_targetConnection.Resolve(ImportXvaAction.RelatedTask)) == null)
|
|
Thread.Sleep(100);
|
|
|
|
// We register a XenObjectsUpdated event handler where we check that the import task has the object creation phase marked as "complete";
|
|
// Once the object creation is complete, we look for the vm; When we found the vm we unregister this event handler;
|
|
m_targetConnection.XenObjectsUpdated += targetConnection_XenObjectsUpdated;
|
|
|
|
Program.Invoke(Program.MainWindow, CheckTask);
|
|
}
|
|
|
|
private void CheckTask()
|
|
{
|
|
Program.AssertOnEventThread();
|
|
|
|
// We need to wait for the task to make the object creation phase as complete
|
|
|
|
if (m_importTask == null || m_importTask.other_config == null)
|
|
return;
|
|
|
|
if (!m_importTask.other_config.ContainsKey("object_creation"))
|
|
return;
|
|
|
|
if (m_importTask.other_config["object_creation"] != "complete")
|
|
return;
|
|
|
|
// Once this is done, look for the vm
|
|
GetVm();
|
|
}
|
|
|
|
private void GetVm()
|
|
{
|
|
Program.AssertOnEventThread();
|
|
|
|
if (m_alreadyFoundVM)
|
|
{
|
|
// Should never get here (as we unregister XenObjectsUpdated event handler after we find the vm) but just in case,
|
|
// unregister XenObjectsUpdated event handler
|
|
m_targetConnection.XenObjectsUpdated -= targetConnection_XenObjectsUpdated;
|
|
return;
|
|
}
|
|
|
|
foreach (VM vm in m_targetConnection.Cache.VMs)
|
|
{
|
|
if (!vm.other_config.ContainsKey(ImportVmAction.IMPORT_TASK))
|
|
continue;
|
|
|
|
string importTaskRef = vm.other_config[ImportVmAction.IMPORT_TASK];
|
|
|
|
if (m_importTask.opaque_ref == importTaskRef)
|
|
{
|
|
m_alreadyFoundVM = true;
|
|
ImportedVm = vm;
|
|
|
|
// We found the VM, unregister XenObjectsUpdated event handler
|
|
m_targetConnection.XenObjectsUpdated -= targetConnection_XenObjectsUpdated;
|
|
|
|
// And close the dialog, flick to next page.
|
|
m_actionDialog.Close();
|
|
if (ImportVmCompleted != null)
|
|
ImportVmCompleted();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Event handlers
|
|
|
|
private void targetConnection_XenObjectsUpdated(object sender, EventArgs e)
|
|
{
|
|
CheckTask();
|
|
}
|
|
|
|
private void m_importXvaAction_Completed(ActionBase sender)
|
|
{
|
|
Program.Invoke(Program.MainWindow, () =>
|
|
{
|
|
Program.AssertOnEventThread();
|
|
|
|
if (!(ImportXvaAction.Succeeded) || ImportXvaAction.Cancelled)
|
|
{
|
|
// task failed or has been cancelled, unregister XenObjectsUpdated event handler
|
|
m_targetConnection.XenObjectsUpdated -= targetConnection_XenObjectsUpdated;
|
|
|
|
// Give the user a chance to correct any errors
|
|
m_actionDialog = null;
|
|
m_buttonPreviousEnabled = true;
|
|
OnPageUpdated();
|
|
}
|
|
});
|
|
}
|
|
|
|
private void m_srPicker_ItemSelectionNotNull(object sender, EventArgs e)
|
|
{
|
|
if (ImportInProgress())
|
|
return;
|
|
|
|
SetButtonNextEnabled(m_srPicker.SR != null);
|
|
IsDirty = true;
|
|
}
|
|
|
|
private void m_srPicker_ItemSelectionNull(object sender, EventArgs e)
|
|
{
|
|
if (ImportInProgress())
|
|
return;
|
|
|
|
SetButtonNextEnabled(m_srPicker.SR != null);
|
|
IsDirty = true;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|