1

我需要使用LoadLibraryGetProcAddress调用某些dll(插入我的应用程序)中的函数,但我不希望它们能够通过包含无限循环或挂起自己来挂起我的程序,所以我正在尝试使用boost::thread(没有经验)来规避这一点。但是,如果某个函数确实挂起,我在调用FreeLibrary清理时会因 ACCESS_DENIED (0x00000005) 而崩溃。

这是重现问题的最小 DLL ( mydll.dll):

extern "C" {
    void __declspec(dllexport) foo() {
        for(;;); //<-- infinite loop
    }
}

这是一个崩溃的最小cpp。为简洁起见,我省略了错误检查,但我可以向您保证不会出现错误:

#include <boost/thread.hpp>
#include <iostream>
#include <windows.h>

int main() {
    using FuncType = void (*)();

    HMODULE dll = LoadLibrary("mydll.dll");

    FuncType func = (FuncType)GetProcAddress(dll, "foo");

    boost::thread thr = [func] {func();};

    if (!thr.timed_join(boost::posix_time::seconds(1))) {
        std::cout << "Timed out.";
        //thr.detach();
        FreeLibrary(dll);
    }

    for (;;);
}

运行时,“超时”。被打印并且崩溃时的调用堆栈(CodeBlocks上的gdb)如下:

#0 00000000 0x6c1c1214 in ??() (??:??)
#1 0040133E operator() (__closure=<optimized out>) (C:\...\thread test.cpp:17)
#5 00000000 0x00000000 in ??() (??:??)

没有优化,就变成了:

#0 00000000 0x6c1c1214 in ??() (??:??)
#1 00401348 operator() (__closure=0x898f90) (C:\...\thread test.cpp:17)
#2 00401780 boost::detail::thread_data<main()::<lambda()> >::run(void)(this=0x898f70) (c:/.../boost/thread/detail/thread.hpp:62)
#5 00000000 0x00000000 in ??() (??:??)

我尝试的第一件事是在thr.detach();上面注释掉的行中添加,但这产生了相同的结果。

我也尝试了以下无济于事:

  • 用函数替换 lambda(我不确定 Boost 与 C++11 的匹配度如何)--- 崩溃
  • 注释掉该FreeLibrary行 --- 工作正常,但我宁愿在加载后释放它,即使它是基于引用的。我也对以后工作的这种行为持谨慎态度。
  • 将 DLL 中的无限循环替换为返回的 --- 工作正常,但我不能指望它
  • 不使用LoadLibrary或任何东西并内联无限循环---工作正常,但我需要使用 DLL:

boost::thread thr = []{for (;;);};

DLL 和 cpp 都是在 Windows 7 64 位上使用 GCC 4.7.1 和 32 位输出类型编译的。我的 Boost 版本是 1.49.0。如果没记错的话,我想在转移到另一台计算机之前,我在 Windows XP 32 位(相同的编译器等)上进行了一两次测试。

为什么会崩溃?如何确保我动态调用的函数不会挂起我的程序?

4

1 回答 1

3

从根本上说,你不能那样做。线程不提供那种隔离。如果需要,请使用流程。

于 2012-09-18T00:18:21.150 回答