stacktrace_windows.cpp 2.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. /*
  2. * Copyright (c) 2012-2015 Daniele Bartolini and individual contributors.
  3. * License: https://github.com/taylor001/crown/blob/master/LICENSE
  4. */
  5. #include "config.h"
  6. #if CROWN_PLATFORM_WINDOWS
  7. #include "log.h"
  8. #include "win_headers.h"
  9. #include <dbghelp.h>
  10. namespace crown
  11. {
  12. void print_callstack()
  13. {
  14. SymInitialize(GetCurrentProcess(), NULL, TRUE);
  15. SymSetOptions(SYMOPT_LOAD_LINES | SYMOPT_UNDNAME);
  16. DWORD mtype;
  17. CONTEXT ctx;
  18. ZeroMemory(&ctx, sizeof(CONTEXT));
  19. ctx.ContextFlags = CONTEXT_CONTROL;
  20. RtlCaptureContext(&ctx);
  21. STACKFRAME64 stack;
  22. ZeroMemory(&stack, sizeof(STACKFRAME64));
  23. #ifdef _M_IX86
  24. mtype = IMAGE_FILE_MACHINE_I386;
  25. stack.AddrPC.Offset = ctx.Eip;
  26. stack.AddrPC.Mode = AddrModeFlat;
  27. stack.AddrFrame.Offset = ctx.Ebp;
  28. stack.AddrFrame.Mode = AddrModeFlat;
  29. stack.AddrStack.Offset = ctx.Esp;
  30. stack.AddrStack.Mode = AddrModeFlat;
  31. #elif _M_X64
  32. mtype = IMAGE_FILE_MACHINE_AMD64;
  33. stack.AddrPC.Offset = ctx.Rip;
  34. stack.AddrPC.Mode = AddrModeFlat;
  35. stack.AddrFrame.Offset = ctx.Rsp;
  36. stack.AddrFrame.Mode = AddrModeFlat;
  37. stack.AddrStack.Offset = ctx.Rsp;
  38. stack.AddrStack.Mode = AddrModeFlat;
  39. #endif
  40. DWORD ldsp = 0;
  41. IMAGEHLP_LINE64 line;
  42. ZeroMemory(&line, sizeof(IMAGEHLP_LINE64));
  43. line.SizeOfStruct = sizeof(IMAGEHLP_LINE64);
  44. char buf[sizeof(SYMBOL_INFO) + MAX_SYM_NAME * sizeof(TCHAR)];
  45. PSYMBOL_INFO sym = (PSYMBOL_INFO)buf;
  46. sym->SizeOfStruct = sizeof(SYMBOL_INFO);
  47. sym->MaxNameLen = MAX_SYM_NAME;
  48. UINT num = 0;
  49. while (StackWalk64(mtype
  50. , GetCurrentProcess()
  51. , GetCurrentThread()
  52. , &stack
  53. , &ctx
  54. , NULL
  55. , SymFunctionTableAccess64
  56. , SymGetModuleBase64
  57. , NULL
  58. ))
  59. {
  60. if (stack.AddrPC.Offset == 0)
  61. break;
  62. ++num;
  63. BOOL res = SymGetLineFromAddr64(GetCurrentProcess()
  64. , stack.AddrPC.Offset
  65. , &ldsp
  66. , &line
  67. );
  68. res = res && SymFromAddr(GetCurrentProcess(), stack.AddrPC.Offset, 0, sym);
  69. if (res == TRUE)
  70. CE_LOGE("\t[%i] %s (%s:%d)", num, sym->Name, line.FileName, line.LineNumber);
  71. else
  72. CE_LOGE("\t[%i] 0x%p", num, stack.AddrPC.Offset);
  73. }
  74. SymCleanup(GetCurrentProcess());
  75. }
  76. } // namespace crown
  77. #endif // CROWN_PLATFORM_WINDOWS