c++ - 使用 C++ 异常时 Windows 上可能的最小堆栈大小(使用 boost 上下文纤维)

标签 c++ windows boost fiber boost-context

我正在使用 boost context 1.67 创建 Fiber(fcontext API) 在 Windows 10 上具有尽可能小的堆栈大小。

这个问题可能不仅特定于 boost 上下文,而且适用于我们使用具有最小堆栈大小的 Windows 线程的任何场景。

我在使用非常小的堆栈(低于 10kb)时遇到了问题 通过由内部引起的 stackoverflow 异常 boost上下文抛出的堆栈展开异常如下所示:

enter image description here

当使用更大的堆栈(> 10 kb)时,我没有遇到任何问题。

对于复制品,以下示例就足够了:

#include <memory>
#include <utility>
#include <boost/context/all.hpp>

#define STACK_SIZE 8000

struct my_allocator
{
  boost::context::stack_context allocate()
  {
    void* memory = std::malloc(STACK_SIZE);
    return {STACK_SIZE,
            static_cast<char*>(memory) +
                STACK_SIZE};
  }

  void deallocate(
      boost::context::stack_context& context)
  {
    std::free(static_cast<char*>(context.sp) -
              STACK_SIZE);
  }
};

int main(int, char**)
{
  boost::context::fiber fiber(
      std::allocator_arg, my_allocator{},
      [](boost::context::fiber&& sink) mutable {
        // ...
        return std::move(sink);
      });

  // Will cause a stack unwind exception and
  // reproduces the issue
  return 0;
}

Boost 上下文仅在此处用于使用用户分配的堆栈执行上下文切换,该问题可能是由于 MSVC C++ 异常的一些限制引起的,这些限制可能需要一定的最小堆栈大小才能工作。此外,SetThreadStackGuarantee WinAPI 函数对这个问题没有任何影响。

堆栈是通过 malloc 分配的,如示例所示。

在 Windows 上使用 C++ 异常时是否可以使用小于 10kb 的堆栈?哪种情况可能导致这里的限制?

最佳答案

不幸的是,Windows API 没有提供返回最小所需堆栈空间的函数或常量。

仅在 32 位 Windows 上,异常 (SEH) 会导致堆栈上的条目。 Win x64 使用基于表的异常处理 - 异常处理程序的条目存储在 pdata 部分中。 因此 x64 上的异常处理不应影响最小值。堆栈空间。 x64 调用约定需要一些空间 - 例如 32 字节的“影子空间”+ XMM 寄存器的空间...但只需要几个字节来存储寄存器。

我猜最小。堆栈空间受到“检测代码”(例如堆栈 cookies 等)的限制 - 可能这可以通过编译器标志来控制。

boost.context分配内存并将其用作堆栈(+实现上下文切换)。 boost. Fiber 使用来自 boost.context 的堆栈,并为每个 Fiber 堆栈顶部的控制结构保留空间(新位置)(但这 < 1kB)。

您可以选择使用间隔嵌套来测试应用程序所需的最小堆栈空间。

关于c++ - 使用 C++ 异常时 Windows 上可能的最小堆栈大小(使用 boost 上下文纤维),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51110650/

相关文章:

c++ - 如何实现 C/C++ 源代码的依赖关系检查

c++ - 使用自定义工具链编译 log4cpp

c++ - getenv ("LINES") 在 Windows 上不起作用

windows - .dll 和 .exe 的区别?

windows - 帮助了解 Windows 内存 - "Working Set"

c++ - Boost:单线程 IO 服务

c++ - BGL 中的 slistS 发生了什么?

c++ - C++ 中的侵入式与非侵入式引用计数指针

c# - 在不同域的远程机器上启动 Windows 服务

c++ - 具有相交多边形边信息的多边形和线段的交集