C#没有捕获非托管C ++ DLL的未处理异常

时间:2010-11-19 09:07:05

标签: c# c++ dll exception-handling pinvoke

我有一个从C#应用程序调用的非托管C ++ dll,我正在尝试使用C#应用程序来捕获所有异常,以便在由于非托管异常而导致dll失败的情况下,然后用户将获得一个不错的错误消息(C#app是一个实现它自己的http处理程序的Web服务)。

我遇到的问题是并非所有类型都被捕获。因此,如果我创建以下内容并执行C#应用程序,则dll会抛出错误并终止整个应用程序。有什么想法吗?

这是在VS2005中使用.Net framework v2

创建的

C ++ - Test.h

#ifndef INC_TEST_H
#define INC_TEST_H

extern "C" __declspec(dllexport) void ProcessBadCall();

#endif

C ++ - Test.cpp

#include <iostream>
#include <vector>

using namespace std;

void ProcessBadCall()
{
  vector<int> myValues;
  int a = myValues[1];
  cout << a << endl;
}

C# - Program.cs

class Program
{
  [DllImport("Test.dll", EntryPoint="ProcessBadCall")]
  static extern void ProcessBadCall();

  static void Main(string[] args)
  {
    try
    {
      ProcessBadCall();
    }
    catch (SEHException ex)
    {
      Console.WriteLine("SEH Exception: {0}", ex.Message);
    }
    catch (Exception ex)
    {
      Console.WriteLine("Exception: {0}", ex.Message);
    }
  }
}

在发布配置下使用以下编译器标志编译dll。

  

/ O2 / GL / D“WIN32”/ D“NDEBUG”/ D.   “_CRT_SECURE_NO_WARNINGS”/ D.   “_UNICODE”/ D“UNICODE”/ D“_WINDLL”   / FD / EHa / MD / Fo“发布”   /Fd"Release\vc80.pdb“/ W4 / WX / nologo   / c / Wp64 / Zi / TP / errorReport:提示

1 个答案:

答案 0 :(得分:5)

尝试使用ExternalException类进行捕获:

http://msdn.microsoft.com/en-us/library/system.runtime.interopservices.externalexception%28v=VS.100%29.aspx

然后,尝试使用异步异常处理(/ EHa)编译非托管C ++ DLL。 看起来你的DLL中有一个读取访问冲突,这是一种异步异常。

AFAIK,只有.NET v4及更高版本默认禁用异步异常的传递。即使这样,您也可以将legacyCorruptedStateExceptionsPolicy = true添加到app.config以启用它。在此之前,它会自动启用(检查是否已在app.config中将其设置为false)。

请注意,我个人认为,无论如何,非托管DLL中的AV本质上都是坏的(并且很危险),而.NET可能只是终止应用程序的正确行为。尝试抛出std :: exception。如果你坚持捕获异步异常,最好的方法是让一个thunking DLL包装try-catch-all围绕调用潜在的bug函数调用。再次,高度/非/推荐(虽然我可以看到它在调试行为不当的DLL时有用)。