【问题标题】:Why after enabling floating point exceptions I got multiple traps error为什么启用浮点异常后出现多个陷阱错误
【发布时间】:2015-11-27 14:37:52
【问题描述】:

使用MSVC2015专业版环境Windows 10,使用/EHa编译

我在做什么:启用浮点异常,以便在发生一些不好的事情时能够捕获异常,仅用于调试

代码:

namespace debug_details
{
void (*defaultStructuredExceptionFunc)(unsigned int, PEXCEPTION_POINTERS) = nullptr;

void my_trans_func(unsigned int exceptionCode, PEXCEPTION_POINTERS pExpInfo)
{
    switch (exceptionCode)
    {
    case EXCEPTION_FLT_DENORMAL_OPERAND:
    case EXCEPTION_FLT_DIVIDE_BY_ZERO:
    case EXCEPTION_FLT_INEXACT_RESULT:
    case EXCEPTION_FLT_INVALID_OPERATION:
    case EXCEPTION_FLT_OVERFLOW:
    case EXCEPTION_FLT_STACK_CHECK:
    case EXCEPTION_FLT_UNDERFLOW:
    {
        _clearfp();
        std::stringstream ss;
        ss << "floating-point structured exception: 0x" << std::hex << exceptionCode;
        throw std::runtime_error(ss.str());
    }
    default:
        if (defaultStructuredExceptionFunc != nullptr)
        {
            defaultStructuredExceptionFunc(exceptionCode, pExpInfo);
        }
    };
};

void EnableFloatingPointExceptions()
{
    unsigned int fe_value = ~(/*_EM_INVALID | _EM_DENORMAL |*/ _EM_ZERODIVIDE | _EM_OVERFLOW | _EM_UNDERFLOW /* | _EM_INEXACT*/);
    unsigned int mask = _MCW_EM;
    unsigned int currentWord = 0;
    _clearfp();
    errno_t result = _controlfp_s(&currentWord, fe_value, mask); // https://msdn.microsoft.com/en-us/library/c9676k6h.aspx
    DVASSERT(result == 0);
    debug_details::defaultStructuredExceptionFunc = _set_se_translator(&debug_details::my_trans_func); // https://msdn.microsoft.com/en-us/library/5z4bw5h5.aspx

    float32 div = 0.f;
    float32 f = 15.f / div;
    float32 f2 = 30.f * div;
}

} // end namespace debug_details

我希望 EXCEPTION_FLT_DIVIDE_BY_ZERO,但有

请帮助了解发生了什么?提前致谢!

【问题讨论】:

    标签: c++ windows math visual-c++


    【解决方案1】:

    我在 MSDN 上找到了答案,x86 平台的问题,在 x64 上,我们可以从线程上下文寄存器中获取正确的异常代码,请参阅参考 - // https://social.msdn.microsoft.com/Forums/en-US/48f63378-19be-413f-88a5-0f24aa72d3c8/the-exceptions-statusfloatmultipletraps-and-statusfloatmultiplefaults-is-needed-in-more

    【讨论】:

      【解决方案2】:

      关于 FP 异常有几点需要了解:

      • controlfp 的标志 mask(即禁用)您设置的异常,而不是启用它们。因此,要仅启用除以 0,您需要在参数中的 EM_ZERODIVIDE 中设置 中的所有内容
      • SSE 代码中的异常不受 controlfp 影响,您需要使用 _mm_setctr(因此,如果您在编译器设置中启用了 SSE/AVX,您的大部分代码可能会使用 SSE,甚至是标量)
      • 如果在被屏蔽时发生异常,稍后启用它会中断(除非某些代码清除了状态字)

      【讨论】:

        猜你喜欢
        • 2019-11-06
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2011-10-15
        • 1970-01-01
        相关资源
        最近更新 更多