10

我创建的 C++/CLI 混合模式 DLL 有问题。当使用它的 .NET 应用程序退出时,它会在卸载时引发异常。执行后DLL_PROCESS_DETACH,DLL 使用自动注册的函数进行运行时清理,atexit() / __onexit()并引发以下异常:

Unhandled exception at 0x752bb9bc (KernelBase.dll) in psstestm.exe: 
0xC0020001: The string binding is invalid.

我已将问题追溯到atexit()由静态提升异常对象注册的调用get_static_exception_object()

    function_to_call    0x0f560410 _t2m@???__Fep@?1???$get_static_exception_object@Ubad_exception_@exception_detail@boost@@@exception_detail@boost@@YA?AVexception_ptr@1@XZ@YAXXZ@?A0x0a546e27@@YAXXZ   void (void)*

我在大多数情况下都使用静态链接的 boost_1_47,除了动态链接的 boost::thread 以避免加载器锁定。我也尝试过动态链接所有没有帮助的提升。#pragma unmanaged此外,所有的提升包括都被块包围。

我希望有人遇到过类似的问题或知道解决方案吗?

谢谢,马克

这是异常发生之前的调用堆栈:

psscorem.dll!_CRT_INIT(void * hDllHandle=0x0f4b0000, unsigned long dwReason=0, void * lpreserved=0x00000001)  Line 413  C
psscorem.dll!__DllMainCRTStartup(void * hDllHandle=0x0f4b0000, unsigned long dwReason=0, void * lpreserved=0x00000001)  Line 526 + 0x11 bytes   C
psscorem.dll!_DllMainCRTStartup(void * hDllHandle=0x0f4b0000, unsigned long dwReason=0, void * lpreserved=0x00000001)  Line 476 + 0x11 bytes    C
mscoreei.dll!__CorDllMain@12()  + 0xde bytes    
mscoree.dll!_ShellShim__CorDllMain@12()  + 0xad bytes   
ntdll.dll!_LdrpCallInitRoutine@16()  + 0x14 bytes   
ntdll.dll!_LdrShutdownProcess@0()  + 0x141 bytes    
ntdll.dll!_RtlExitUserProcess@4()  + 0x74 bytes 
kernel32.dll!749479f5()     
mscoreei.dll!RuntimeDesc::ShutdownAllActiveRuntimes()  + 0xc8 bytes 
mscoreei.dll!CLRRuntimeHostInternalImpl::ShutdownAllRuntimesThenExit()  + 0x15 bytes    
clr.dll!EEPolicy::ExitProcessViaShim()  + 0x66 bytes    
clr.dll!SafeExitProcess()  + 0x99 bytes 
clr.dll!DisableRuntime()  - 0x1146bb bytes  
clr.dll!EEPolicy::HandleExitProcess()  + 0x57 bytes 
clr.dll!__CorExeMainInternal@0()  + 0x11c bytes 
clr.dll!__CorExeMain@0()  + 0x1c bytes  
mscoreei.dll!__CorExeMain@0()  + 0x38 bytes 
mscoree.dll!_ShellShim__CorExeMain@0()  + 0x227 bytes   
mscoree.dll!__CorExeMain_Exported@0()  + 0x8 bytes  
kernel32.dll!@BaseThreadInitThunk@12()  + 0x12 bytes    
ntdll.dll!___RtlUserThreadStart@8()  + 0x27 bytes   
ntdll.dll!__RtlUserThreadStart@8()  + 0x1b bytes    
4

3 回答 3

10

我遇到了同样的问题,并设法将其追踪到 exception_ptr.hpp 中的以下函数:

    template <class Exception>
    exception_ptr
    get_static_exception_object()
        {
        Exception ba;
        exception_detail::clone_impl<Exception> c(ba);
        c <<
            throw_function(BOOST_CURRENT_FUNCTION) <<
            throw_file(__FILE__) <<
            throw_line(__LINE__);
        static exception_ptr ep(shared_ptr<exception_detail::clone_base const>(new exception_detail::clone_impl<Exception>(c)));
        return ep;
        }

这里有问题的部分是:static exception_ptr ep(...

您可以删除静态,它应该可以工作:

    template <class Exception>
    exception_ptr
    get_static_exception_object()
        {
        Exception ba;
        exception_detail::clone_impl<Exception> c(ba);
        c <<
            throw_function(BOOST_CURRENT_FUNCTION) <<
            throw_file(__FILE__) <<
            throw_line(__LINE__);
        exception_ptr ep(shared_ptr<exception_detail::clone_base const>(new exception_detail::clone_impl<Exception>(c)));
        return ep;
        }

注意这个函数是如何使用的,它将返回的静态变量分配给另一个静态变量。这个函数的整个实现看起来很可疑,可能我会提出一个关于提升支持的问题。

可能有其他解决方法可以解决此问题。更多关于混合程序集中静态变量的分析可以在这里找到:http: //derevyanko.blogspot.com/2009/01/clic.html但是只有俄语版本。

于 2012-02-22T12:26:32.573 回答
0

Boost 邮件列表中的一篇文章所述,一种方法是将托管代码和非托管代码分成单独的翻译单元(.cpp 文件和它们#include 的标头)。仅从非托管翻译单元引用 Boost。仅为托管翻译单元打开 /clr。

于 2013-01-14T15:26:46.197 回答
0

您可以添加以下行:

#if _MANAGED
#error "Don't include that file in CLI compilation units. It will cause failure when cleaning the static objects of the managed dll"
#endif

在 get_static_exception_object 声明之前,并且不在您的 cli 文件中仅包含该文件(或包含该文件的 boost 标头)。

boost/thread.hpp对我来说,通过boost/thread/thread.hpp解决问题替换一个。

于 2016-04-05T09:49:08.603 回答