3

我基本上有一个重复的问题,但对于 .NET Core。

我有核心控制台应用程序:

class Program
{
    static void DoSomeAccessViolation()
    {
        // if you have any questions about why this throws,
        // the answer is "42", of course

        var ptr = new IntPtr(42);
        Marshal.StructureToPtr(42, ptr, true);
    }

    [SecurityCritical]
    [HandleProcessCorruptedStateExceptions]
    static void Main(string[] args)
    {
        try
        {
            DoSomeAccessViolation();
        }
        catch (Exception ex)
        {
            Console.Error.WriteLine(ex);
        }
    }
}

我试图添加一个 Settings.setting 文件

<SettingsFile xmlns="http://schemas.microsoft.com/VisualStudio/2004/01/settings" CurrentProfile="(Default)" GeneratedClassNamespace="ConsoleApp2" GeneratedClassName="Settings1">
  <Profiles />
  <Settings>
    <Setting Name="legacyCorruptedStateExceptionsPolicy" Type="System.Boolean" Scope="Application">
      <Value Profile="(Default)">True</Value>
    </Setting>
  </Settings>
</SettingsFile>

我尝试根据文档设置环境变量:

C:\>set COMPlus_legacyCorruptedStateExceptionsPolicy=1
C:\>dotnet run

或者

C:\>set COMPlus_legacyCorruptedStateExceptionsPolicy=true
C:\>dotnet run

(在 Windows 上)。但是没有任何效果,应用程序总是在不打印异常的情况下严重崩溃。

4

3 回答 3

2

我一直在研究这个主题的一些背景,按时间顺序......

dotnet/coreclr问题 #9045:剥离损坏状态异常处理

dotnet/coreclr PR #10957:不尊重属性 HandleProcessCorruptedStateExceptions

dotnet/coreclr问题 #19192:无法在托管代码中捕获本机 Linux 代码引发的任何用户异常

我现在是 SOL。必须创建各种非托管包装器来捕获外部 CSE。

您可以尝试更多选项,特别是结合 legacyCorruptedStateExceptionsPolicy 的 FailFastOnCorruptedStateException。

于 2019-01-07T04:05:59.840 回答
1

我运行了您的代码,引发的异常是 ExecutionEngineException。这是运行时中的致命错误,意味着运行时无法进一步执行您的程序。您无法捕获此异常是有充分理由的:您无法处理它。您的代码无法“修复”运行时并继续运行。

来自 Microsoft 文档:当公共语言运行时的执行引擎出现内部错误时引发的异常。并且:此类型之前表示运行时中存在未指定的致命错误。运行时不再引发此异常,因此此类型已过时。

于 2018-02-08T10:09:54.353 回答
0

.net 核心中唯一保留的就是legacyCorruptedStateExceptionsPolicy行为。

要启用此功能,您必须在进程开始之前将COMPlus_legacyCorruptedStateExceptionsPolicy环境变量设置为。1但是,在 .net 核心中,由于在 .net 移植期间做出的决定,它捕获的损坏状态异常比在 .net 中要少得多:(

例如,如果您在代码中替换为Marshal.StructureToPtr(42, (IntPtr)42, true);Marshal.ReadInt32((IntPtr)42);并设置环境变量)AVE 将成功生成并捕获。

您可以在此处阅读有关此问题的详细信息。

总体而言,关于 .net 核心中的损坏状态异常:

  1. 要捕获其中的一些,请在进程开始之前将COMPlus_legacyCorruptedStateExceptionsPolicy环境变量设置为。1
  2. 如果您从本地调用中获得 CSE。尝试将异常处理移到那里(如果您有源代码),或者按照 TylerY86 的建议尝试非托管包装器。
  3. 编写 WatchDog 进程来检查您的应用程序崩溃(如果它很重要)。然后查看事件日志并分析故障转储...
于 2019-11-19T13:09:17.273 回答