Skip to content

Commit

Permalink
Debug: print call stack of unhandled exceptions
Browse files Browse the repository at this point in the history
  • Loading branch information
SChernykh committed Nov 7, 2023
1 parent e92ce0a commit 8699e18
Show file tree
Hide file tree
Showing 3 changed files with 70 additions and 9 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/cppcheck.yml
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ on:
jobs:
cppcheck-ubuntu:

timeout-minutes: 50
timeout-minutes: 60
runs-on: ubuntu-latest

steps:
Expand Down Expand Up @@ -60,7 +60,7 @@ jobs:

cppcheck-windows:

timeout-minutes: 50
timeout-minutes: 60
runs-on: windows-latest

steps:
Expand Down
69 changes: 67 additions & 2 deletions src/log.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -44,8 +44,64 @@ bool CONSOLE_COLORS = true;
#ifdef _WIN32
static const HANDLE hStdIn = GetStdHandle(STD_INPUT_HANDLE);
static const HANDLE hStdOut = GetStdHandle(STD_OUTPUT_HANDLE);
static const HANDLE hStdErr = GetStdHandle(STD_ERROR_HANDLE);
#endif

#if defined(_MSC_VER) && !defined(NDEBUG)

#include <DbgHelp.h>

#pragma comment(lib, "Dbghelp.lib")

LONG WINAPI UnhandledExceptionFilter(_In_ _EXCEPTION_POINTERS*)
{
constexpr size_t MAX_FRAMES = 32;

void* stack_trace[MAX_FRAMES] = {};
DWORD hash;
CaptureStackBackTrace(1, MAX_FRAMES, stack_trace, &hash);

char buffer[sizeof(SYMBOL_INFO) + MAX_SYM_NAME * sizeof(TCHAR)] = {};
PSYMBOL_INFO pSymbol = reinterpret_cast<PSYMBOL_INFO>(buffer);

pSymbol->SizeOfStruct = sizeof(SYMBOL_INFO);
pSymbol->MaxNameLen = MAX_SYM_NAME;

IMAGEHLP_LINE64 line{};
line.SizeOfStruct = sizeof(IMAGEHLP_LINE64);

const HANDLE h = GetCurrentProcess();

fprintf(stderr, "\n\nUnhandled exception at:\n");
fflush(stderr);

for (size_t j = 0; j < MAX_FRAMES; ++j) {
const DWORD64 address = reinterpret_cast<DWORD64>(stack_trace[j]);
DWORD t = 0;
if (SymFromAddr(h, address, nullptr, pSymbol) && SymGetLineFromAddr64(h, address, &t, &line)) {
fprintf(stderr, "%s (%s, line %lu)\n", line.FileName, pSymbol->Name, line.LineNumber);
fflush(stderr);
}
}

fprintf(stderr, "\n\n");
fflush(stderr);

// Normal logging might be broken at this point, but try to log it anyway
LOGERR(0, "Unhandled exception at:");

for (size_t j = 0; j < MAX_FRAMES; ++j) {
const DWORD64 address = reinterpret_cast<DWORD64>(stack_trace[j]);
DWORD t = 0;
if (SymFromAddr(h, address, nullptr, pSymbol) && SymGetLineFromAddr64(h, address, &t, &line)) {
LOGERR(0, line.FileName << " (" << static_cast<const char*>(pSymbol->Name) << ", line " << static_cast<size_t>(line.LineNumber) << ')');
}
}

Sleep(1000);

return EXCEPTION_CONTINUE_SEARCH;
}
#endif // _MSC_VER && !NDEBUG
#endif // _WIN32

class Worker
{
Expand All @@ -62,6 +118,11 @@ class Worker
, m_started{ false }
, m_stopped(false)
{
#if defined(_WIN32) && defined(_MSC_VER) && !defined(NDEBUG)
SetUnhandledExceptionFilter(UnhandledExceptionFilter);
SymInitialize(GetCurrentProcess(), NULL, TRUE);
#endif

set_main_thread();

std::setlocale(LC_ALL, "en_001");
Expand Down Expand Up @@ -140,6 +201,10 @@ class Worker
#endif

m_logFile.close();

#if defined(_WIN32) && defined(_MSC_VER) && !defined(NDEBUG)
SymCleanup(GetCurrentProcess());
#endif
}

FORCEINLINE void write(const char* buf, uint32_t size)
Expand Down
6 changes: 1 addition & 5 deletions src/memory_leak_debug.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
#include "common.h"

// Simple memory leak detector for Windows users, works best in RelWithDebInfo configuration.
#if defined(_WIN32) && defined(DEV_TRACK_MEMORY)
#if defined(_WIN32) && defined(DEV_TRACK_MEMORY) && defined(_MSC_VER) && !defined(NDEBUG)

#include "uv_util.h"
#include <atomic>
Expand Down Expand Up @@ -270,8 +270,6 @@ void memory_tracking_start()
// Trigger std::ostream initialization to avoid reporting it as leaks
std::cout << "Memory leak detection = " << 1 << std::endl;

SymInitialize(GetCurrentProcess(), NULL, TRUE);

using namespace p2pool;

uv_replace_allocator(malloc_hook, realloc_hook, calloc_hook, free_hook);
Expand Down Expand Up @@ -307,8 +305,6 @@ bool memory_tracking_stop()
printf("No memory leaks detected\n\n");
}

SymCleanup(h);

return (total_leaks == 0);
}

Expand Down

0 comments on commit 8699e18

Please sign in to comment.