RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 788394
Accepted
LmTinyToon
LmTinyToon
Asked:2020-02-21 23:03:09 +0000 UTC2020-02-21 23:03:09 +0000 UTC 2020-02-21 23:03:09 +0000 UTC

如何在异常处理程序中打印调用堆栈?

  • 772

我在 Windows 上有一个 c++ 程序,我想在其中提供完整的调用堆栈日志记录,以防出现错误(硬件、软件)。是否有可能实现这样的功能?这是包含外部 seh 处理程序的一小段代码:

__try
{
     difficult_task();
}
__except(my_seh_filter(GetExceptionInformation()))
{
    // How to print full backtrace of exception here?
}

我目前所知道的:

  1. Windows 提供的 DbgHelp.dll 允许您获取有关调用堆栈的信息(您可以打印堆栈)。
  2. SEH 异常处理允许您获取发生异常的上下文。(异常类型、异常地址、抛出异常时的处理器寄存器)。

有一个但是。我想直接在 seh 过滤器或相应的 catch 块中打印调用堆栈。而且我并不完全清楚如何使用这两个工具打印回溯(毕竟,它可能会在堆栈提升时发生变化)。当我在抛出异常的那一刻打印堆栈时,这是一回事。另一个是在处理时(我假设在处理异常时,我将不再拥有有关堆栈的完整信息)。

c++
  • 1 1 个回答
  • 10 Views

1 个回答

  • Voted
  1. Best Answer
    MSDN.WhiteKnight
    2020-04-05T16:56:07Z2020-04-05T16:56:07Z

    从 SEH 过滤器中取出调用堆栈的最简单方法是将指向由 GetExceptionInformation返回的上下文的指针传递给 StackWalk64函数。像这样的东西:

    #include <stdlib.h>
    #include <locale.h>
    #include <stdio.h>
    #include <Windows.h>
    #include "DbgHelp.h"
    #pragma comment(lib, "Dbghelp.lib")
    
    const int MaxNameLen = 256;
    
    //выводит стек вызовов, соответствующий указанному контексту
    void printStack( CONTEXT* ctx )
    {
        BOOL    result;
        HANDLE  process;
        HANDLE  thread;
        HMODULE hModule;
                
        STACKFRAME64        stack;
        ULONG               frame;    
        DWORD64             displacement;
    
        char buffer[sizeof(SYMBOL_INFO) + MAX_SYM_NAME * sizeof(TCHAR)];
        char name[ MaxNameLen ];
        char module[MaxNameLen];
        PSYMBOL_INFO pSymbol = (PSYMBOL_INFO)buffer;
    
        // StackWalk64 модифицирует контекст в 64-битных приложениях,
        // что может привести к падению. Для предотвращения этого  
        // создадим копию контекста.
        CONTEXT ctxCopy;
        memcpy(&ctxCopy, ctx, sizeof(CONTEXT));
        
        memset( &stack, 0, sizeof( STACKFRAME64 ) );
    
        process                = GetCurrentProcess();
        thread                 = GetCurrentThread();
        displacement           = 0;
    #if !defined(_M_AMD64)
        stack.AddrPC.Offset    = (*ctx).Eip;
        stack.AddrPC.Mode      = AddrModeFlat;
        stack.AddrStack.Offset = (*ctx).Esp;
        stack.AddrStack.Mode   = AddrModeFlat;
        stack.AddrFrame.Offset = (*ctx).Ebp;
        stack.AddrFrame.Mode   = AddrModeFlat;
    #endif
    
        SymInitialize( process, NULL, TRUE ); //загружаем символы
    
        for( frame = 0; ; frame++ )
        {
            //получаем следующий вызов из стека
            result = StackWalk64
            (
    #if defined(_M_AMD64)
                IMAGE_FILE_MACHINE_AMD64
    #else
                IMAGE_FILE_MACHINE_I386
    #endif
                ,
                process,
                thread,
                &stack,
                &ctxCopy,
                NULL,
                SymFunctionTableAccess64,
                SymGetModuleBase64,
                NULL
            );
    
            if( !result ) break;        
    
            //полчаем имя символа для адреса
            pSymbol->SizeOfStruct = sizeof(SYMBOL_INFO);
            pSymbol->MaxNameLen = MAX_SYM_NAME;
            SymFromAddr(process, ( ULONG64 )stack.AddrPC.Offset, &displacement, pSymbol);
                    
            hModule = NULL;
            lstrcpyA(module,"");        
            GetModuleHandleEx(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT, 
                (LPCTSTR)(stack.AddrPC.Offset), &hModule);
    
            //получаем имя модуля
            if(hModule != NULL)GetModuleFileNameA(hModule,module,MaxNameLen);       
    
            printf ("%lu: %s  (in %s)\n",frame,pSymbol->Name,module);       
                    
        }
    }
    
    
    
    void BuggedFunction()
    {
        int * p = NULL;
        *p = 5;
    }
    
    void bar()
    {
        BuggedFunction();
    }
    
    void foo()
    {    
         bar();    
    }
    
    int seh_filter(_EXCEPTION_POINTERS* ex)
    {
        printf("Exception 0x%x occured\n",ex->ExceptionRecord->ExceptionCode);  
        printStack(ex->ContextRecord);
        
        return EXCEPTION_EXECUTE_HANDLER;
    }
    
    int main(void)
    {
        setlocale(LC_ALL,"Russian");
        
        __try
        {
            foo();
        }
        __except(seh_filter(GetExceptionInformation()))
        {       
            printf("Exception \n");         
        }
        getchar();
        return 0;
    }
    
    • 2

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    是否可以在 C++ 中继承类 <---> 结构?

    • 2 个回答
  • Marko Smith

    这种神经网络架构适合文本分类吗?

    • 1 个回答
  • Marko Smith

    为什么分配的工作方式不同?

    • 3 个回答
  • Marko Smith

    控制台中的光标坐标

    • 1 个回答
  • Marko Smith

    如何在 C++ 中删除类的实例?

    • 4 个回答
  • Marko Smith

    点是否属于线段的问题

    • 2 个回答
  • Marko Smith

    json结构错误

    • 1 个回答
  • Marko Smith

    ServiceWorker 中的“获取”事件

    • 1 个回答
  • Marko Smith

    c ++控制台应用程序exe文件[重复]

    • 1 个回答
  • Marko Smith

    按多列从sql表中选择

    • 1 个回答
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Suvitruf - Andrei Apanasik 什么是空? 2020-08-21 01:48:09 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5