使用XP Pro上的WebBrowser控件访问违规绘画ToolStripComboBox

时间:2012-09-02 16:07:54

标签: .net windows

我正在调试一个问题,当用户锁定/解锁计算机或用户点击ctrl-alt-delete然后命中转义时,.NET 4.0(WinForm,而不是WPF)应用程序在Windows XP上崩溃必须锁定在这种情况下 - 但他们可以选择锁定,启动任务管理器等)。这是非常可重复的。

这与绘制ToolStripComboBox有关。这会在引擎盖下的一些AccessViolationException例程中生成gdiplus

有几种不同的方式让我看到它崩溃,但所有这些都在绘制这个控件的同一区域。这是一个堆栈跟踪:

System.AccessViolationException: Attempted to read or write protected memory. This is often an indication that other memory is corrupt.
at System.Drawing.SafeNativeMethods.Gdip.GdipFillRectangleI(HandleRef graphics, HandleRef brush, Int32 x, Int32 y, Int32 width, Int32 height)
at System.Drawing.Graphics.FillRectangle(Brush brush, Int32 x, Int32 y, Int32 width, Int32 height)
at System.Drawing.Graphics.FillRectangle(Brush brush, Rectangle rect)
at System.Windows.Forms.ToolStripComboBox.ToolStripComboBoxControl.ToolStripComboBoxFlatComboAdapter.DrawFlatComboDropDown(ComboBox comboBox, Graphics g, Rectangle dropDownRect)
at System.Windows.Forms.ComboBox.FlatComboAdapter.DrawFlatCombo(ComboBox comboBox, Graphics g)
at System.Windows.Forms.ComboBox.WndProc(Message& m)
at System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
at System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
...

有没有人有关于如何解决这个问题的建议,或者机器锁定/解锁或ctrl-alt-delete屏幕的重要性是什么?

编辑:

我把它归结为下面粘贴的一个简单的应用程序,它在XP Pro上非常可重复。这很香草 - 就像我们能得到的一样简单。一切都是在UI线程上创建/操作的。

namespace Test
{
    static class Program
    {
        /// <summary>
        /// The main entry point for the application.
        /// </summary>
        [STAThread]
        static void Main()
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(false);
            Application.Run(new Form1());
        }
    }

    public class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, System.EventArgs e)
        {
            webBrowser2.Navigate("http://www.cnn.com");
        }

        /// <summary>
        /// Required designer variable.
        /// </summary>
        private System.ComponentModel.IContainer components = null;

        /// <summary>
        /// Clean up any resources being used.
        /// </summary>
        /// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param>
        protected override void Dispose(bool disposing)
        {
            if (disposing && (components != null))
            {
                components.Dispose();
            }
            base.Dispose(disposing);
        }

        #region Windows Form Designer generated code

        /// <summary>
        /// Required method for Designer support - do not modify
        /// the contents of this method with the code editor.
        /// </summary>
        private void InitializeComponent()
        {
            this.toolStripContainer1 = new System.Windows.Forms.ToolStripContainer();
            this.webBrowser2 = new System.Windows.Forms.WebBrowser();
            this.toolStrip1 = new System.Windows.Forms.ToolStrip();
            this.toolStripLabel1 = new System.Windows.Forms.ToolStripLabel();
            this.toolStripComboBox1 = new System.Windows.Forms.ToolStripComboBox();
            this.toolStripContainer1.ContentPanel.SuspendLayout();
            this.toolStripContainer1.TopToolStripPanel.SuspendLayout();
            this.toolStripContainer1.SuspendLayout();
            this.toolStrip1.SuspendLayout();
            this.SuspendLayout();
            // 
            // toolStripContainer1
            // 
            this.toolStripContainer1.BottomToolStripPanelVisible = false;
            // 
            // toolStripContainer1.ContentPanel
            // 
            this.toolStripContainer1.ContentPanel.Controls.Add(this.webBrowser2);
            this.toolStripContainer1.ContentPanel.Size = new System.Drawing.Size(555, 268);
            this.toolStripContainer1.Dock = System.Windows.Forms.DockStyle.Fill;
            this.toolStripContainer1.LeftToolStripPanelVisible = false;
            this.toolStripContainer1.Location = new System.Drawing.Point(0, 0);
            this.toolStripContainer1.Name = "toolStripContainer1";
            this.toolStripContainer1.RightToolStripPanelVisible = false;
            this.toolStripContainer1.Size = new System.Drawing.Size(555, 296);
            this.toolStripContainer1.TabIndex = 1;
            this.toolStripContainer1.Text = "toolStripContainer1";
            // 
            // toolStripContainer1.TopToolStripPanel
            // 
            this.toolStripContainer1.TopToolStripPanel.Controls.Add(this.toolStrip1);
            // 
            // webBrowser2
            // 
            this.webBrowser2.Dock = System.Windows.Forms.DockStyle.Fill;
            this.webBrowser2.Location = new System.Drawing.Point(0, 0);
            this.webBrowser2.MinimumSize = new System.Drawing.Size(20, 20);
            this.webBrowser2.Name = "webBrowser2";
            this.webBrowser2.Size = new System.Drawing.Size(555, 268);
            this.webBrowser2.TabIndex = 0;
            // 
            // toolStrip1
            // 
            this.toolStrip1.Dock = System.Windows.Forms.DockStyle.None;
            this.toolStrip1.Items.AddRange(new System.Windows.Forms.ToolStripItem[] {
            this.toolStripLabel1,
            this.toolStripComboBox1});
            this.toolStrip1.Location = new System.Drawing.Point(3, 0);
            this.toolStrip1.Name = "toolStrip1";
            this.toolStrip1.Size = new System.Drawing.Size(173, 28);
            this.toolStrip1.TabIndex = 0;
            // 
            // toolStripLabel1
            // 
            this.toolStripLabel1.Name = "toolStripLabel1";
            this.toolStripLabel1.Size = new System.Drawing.Size(38, 25);
            this.toolStripLabel1.Text = "blah";
            // 
            // toolStripComboBox1
            // 
            this.toolStripComboBox1.DropDownStyle = System.Windows.Forms.ComboBoxStyle.DropDownList;
            this.toolStripComboBox1.Items.AddRange(new object[] {
            "a b",
            "c",
            "d",
            "e",
            "f"});
            this.toolStripComboBox1.Name = "toolStripComboBox1";
            this.toolStripComboBox1.Size = new System.Drawing.Size(121, 28);
            // 
            // Form1
            // 
            this.AutoScaleDimensions = new System.Drawing.SizeF(8F, 16F);
            this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
            this.ClientSize = new System.Drawing.Size(555, 296);
            this.Controls.Add(this.toolStripContainer1);
            this.Name = "Form1";
            this.Text = "Form1";
            this.Load += new System.EventHandler(this.Form1_Load);
            this.toolStripContainer1.ContentPanel.ResumeLayout(false);
            this.toolStripContainer1.TopToolStripPanel.ResumeLayout(false);
            this.toolStripContainer1.TopToolStripPanel.PerformLayout();
            this.toolStripContainer1.ResumeLayout(false);
            this.toolStripContainer1.PerformLayout();
            this.toolStrip1.ResumeLayout(false);
            this.toolStrip1.PerformLayout();
            this.ResumeLayout(false);
        }
        #endregion

        private ToolStripContainer toolStripContainer1;
        private ToolStrip toolStrip1;
        private ToolStripLabel toolStripLabel1;
        private ToolStripComboBox toolStripComboBox1;
        private WebBrowser webBrowser2;
    }
}

编辑2:

我无法在Windows XP Media Center版本上重现这一点,目前只有XP专业版。

1 个答案:

答案 0 :(得分:1)

我们已经有一段时间了。相同的调用堆栈,当用户执行ctrl-alt-del并转义时发生。 场景有点不同,它发生在我们的WinForm应用程序中,它是WinForm和WPF控件的混合。在应用程序中,我们有一个在ElementHost中托管的WPF UserControl,当此组件位于浮动窗口中时,将鼠标悬停在重新绘制的WinForm控件上,并且屏幕被锁定和解锁,将引发AccessViolationException并且应用程序崩溃。我应该提一下,我们也在使用Infragistics UltraDockWorkspace。

我们还没有找到解决方案,但是在最近使用MS源进行调试时,我注意到在抛出异常的方法中的这个注释(在System.Drawing中的Graphics.cs中):

         /// <devdoc>
    ///     GDI+ will return a 'generic error' with specific win32 last error codes when
    ///     a terminal server session has been closed, minimized, etc...  We don't want 
    ///     to throw when this happens, so we'll guard against this by looking at the
    ///     'last win32 error code' and checking to see if it is either 1) access denied 
    ///     or 2) proc not found and then ignore it. 
    ///
    ///     The problem is that when you lock the machine, the secure desktop is enabled and 
    ///     rendering fails which is expected (since the app doesn't have permission to draw
    ///     on the secure desktop). Not sure if there's anything you can do, short of catching
    ///     the desktop switch message and absorbing all the exceptions that get thrown while
    ///     it's the secure desktop. 
    /// </devdoc>
    private void CheckErrorStatus(int status) { 
        if (status != SafeNativeMethods.Gdip.Ok) { 
            // Generic error from GDI+ can be GenericError or Win32Error.
            if (status == SafeNativeMethods.Gdip.GenericError || status == SafeNativeMethods.Gdip.Win32Error) { 
                int error = Marshal.GetLastWin32Error();
                if (error == SafeNativeMethods.ERROR_ACCESS_DENIED || error == SafeNativeMethods.ERROR_PROC_NOT_FOUND ||
                        //here, we'll check to see if we are in a term. session...
                        (((UnsafeNativeMethods.GetSystemMetrics(NativeMethods.SM_REMOTESESSION) & 0x00000001) != 0) && (error == 0))) { 
                        return;
                    } 
                } 

            //legitimate error, throw our status exception 
            throw SafeNativeMethods.Gdip.StatusException(status);
        }
    }

据我所知,由于在安全桌面模式下进行渲染,因此应该在此处强制使用AccessViolationException,但在某些情况下,会传播此异常。

抱歉,我无法提供解决方案,但希望此信息有用。