Add stackwalk64 check/fallback
This commit is contained in:
parent
4324901cd1
commit
b690d3805d
@ -97,13 +97,12 @@ if(NOT CMAKE_CXX_COMPILER_ID STREQUAL "MSVC")
|
|||||||
check_support(HAS_CXXABI has_cxxabi.cpp "" "" "")
|
check_support(HAS_CXXABI has_cxxabi.cpp "" "" "")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(NOT WIN32) # No need to bother checking in msvc, but do check in minngw
|
if(NOT WIN32)
|
||||||
check_support(HAS_UNWIND has_unwind.cpp "" "" "")
|
check_support(HAS_UNWIND has_unwind.cpp "" "" "")
|
||||||
check_support(HAS_EXECINFO has_execinfo.cpp "" "" "")
|
check_support(HAS_EXECINFO has_execinfo.cpp "" "" "")
|
||||||
check_support(HAS_BACKTRACE has_backtrace.cpp "" "backtrace" "${CPPTRACE_BACKTRACE_PATH_DEFINITION}")
|
check_support(HAS_BACKTRACE has_backtrace.cpp "" "backtrace" "${CPPTRACE_BACKTRACE_PATH_DEFINITION}")
|
||||||
check_support(HAS_DL has_dl.cpp "" "dl" "")
|
check_support(HAS_DL has_dl.cpp "" "dl" "")
|
||||||
set(STACKTRACE_LINK_LIB "stdc++_libbacktrace")
|
set(STACKTRACE_LINK_LIB "stdc++_libbacktrace")
|
||||||
check_support(HAS_STACKTRACE has_stacktrace.cpp "" "${STACKTRACE_LINK_LIB}" "")
|
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
find_program(ADDR2LINE_PATH atos PATHS ENV PATH)
|
find_program(ADDR2LINE_PATH atos PATHS ENV PATH)
|
||||||
else()
|
else()
|
||||||
@ -114,6 +113,8 @@ if(NOT WIN32) # No need to bother checking in msvc, but do check in minngw
|
|||||||
else()
|
else()
|
||||||
set(HAS_ADDR2LINE TRUE)
|
set(HAS_ADDR2LINE TRUE)
|
||||||
endif()
|
endif()
|
||||||
|
else()
|
||||||
|
check_support(HAS_STACKWALK has_stackwalk.cpp "" "dbghelp" "")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# =============================================== Autoconfig unwinding ===============================================
|
# =============================================== Autoconfig unwinding ===============================================
|
||||||
@ -139,12 +140,14 @@ if(
|
|||||||
set(CPPTRACE_UNWIND_WITH_NOTHING On)
|
set(CPPTRACE_UNWIND_WITH_NOTHING On)
|
||||||
message(FATAL_ERROR "Cpptrace auto config: No unwinding back-end seems to be supported, stack tracing will not work. To compile anyway set CPPTRACE_UNWIND_WITH_NOTHING.")
|
message(FATAL_ERROR "Cpptrace auto config: No unwinding back-end seems to be supported, stack tracing will not work. To compile anyway set CPPTRACE_UNWIND_WITH_NOTHING.")
|
||||||
endif()
|
endif()
|
||||||
elseif(MINGW)
|
elseif(MINGW OR WIN32)
|
||||||
set(CPPTRACE_UNWIND_WITH_DBGHELP On)
|
if(HAS_STACKWALK)
|
||||||
message(STATUS "Cpptrace auto config: Using dbghelp for unwinding")
|
set(CPPTRACE_UNWIND_WITH_DBGHELP On)
|
||||||
elseif(WIN32)
|
message(STATUS "Cpptrace auto config: Using dbghelp for unwinding")
|
||||||
set(CPPTRACE_UNWIND_WITH_DBGHELP On)
|
else()
|
||||||
message(STATUS "Cpptrace auto config: Using dbghelp for unwinding")
|
set(CPPTRACE_UNWIND_WITH_WINAPI On)
|
||||||
|
message(STATUS "Cpptrace auto config: Using winapi for unwinding")
|
||||||
|
endif()
|
||||||
endif()
|
endif()
|
||||||
else()
|
else()
|
||||||
#message(STATUS "MANUAL CONFIG SPECIFIED")
|
#message(STATUS "MANUAL CONFIG SPECIFIED")
|
||||||
|
|||||||
@ -1,8 +0,0 @@
|
|||||||
#include <stacktrace>
|
|
||||||
|
|
||||||
int main() {
|
|
||||||
std::stacktrace trace = std::stacktrace::current();
|
|
||||||
for(const auto entry : trace) {
|
|
||||||
(void)entry;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
101
cmake/has_stackwalk.cpp
Normal file
101
cmake/has_stackwalk.cpp
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
#include <dbghelp.h>
|
||||||
|
|
||||||
|
#define IS_CLANG 0
|
||||||
|
#define IS_GCC 0
|
||||||
|
#define IS_MSVC 0
|
||||||
|
|
||||||
|
#if defined(__clang__)
|
||||||
|
#undef IS_CLANG
|
||||||
|
#define IS_CLANG 1
|
||||||
|
#elif defined(__GNUC__) || defined(__GNUG__)
|
||||||
|
#undef IS_GCC
|
||||||
|
#define IS_GCC 1
|
||||||
|
#elif defined(_MSC_VER)
|
||||||
|
#undef IS_MSVC
|
||||||
|
#define IS_MSVC 1
|
||||||
|
#else
|
||||||
|
#error "Unsupported compiler"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int main() {
|
||||||
|
HANDLE proc = GetCurrentProcess();
|
||||||
|
HANDLE thread = GetCurrentThread();
|
||||||
|
// https://jpassing.com/2008/03/12/walking-the-stack-of-the-current-thread/
|
||||||
|
|
||||||
|
// Get current thread context
|
||||||
|
// GetThreadContext cannot be used on the current thread.
|
||||||
|
// RtlCaptureContext doesn't work on i386
|
||||||
|
CONTEXT context;
|
||||||
|
#if defined(_M_IX86) || defined(__i386__)
|
||||||
|
ZeroMemory(&context, sizeof(CONTEXT));
|
||||||
|
context.ContextFlags = CONTEXT_CONTROL;
|
||||||
|
#if IS_MSVC
|
||||||
|
__asm {
|
||||||
|
label:
|
||||||
|
mov [context.Ebp], ebp;
|
||||||
|
mov [context.Esp], esp;
|
||||||
|
mov eax, [label];
|
||||||
|
mov [context.Eip], eax;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
asm(
|
||||||
|
"label:\n\t"
|
||||||
|
"mov{l %%ebp, %[cEbp] | %[cEbp], ebp};\n\t"
|
||||||
|
"mov{l %%esp, %[cEsp] | %[cEsp], esp};\n\t"
|
||||||
|
"mov{l $label, %%eax | eax, label};\n\t"
|
||||||
|
"mov{l %%eax, %[cEip] | %[cEip], eax};\n\t"
|
||||||
|
: [cEbp] "=r" (context.Ebp),
|
||||||
|
[cEsp] "=r" (context.Esp),
|
||||||
|
[cEip] "=r" (context.Eip)
|
||||||
|
);
|
||||||
|
#endif
|
||||||
|
#else
|
||||||
|
RtlCaptureContext(&context);
|
||||||
|
#endif
|
||||||
|
// Setup current frame
|
||||||
|
STACKFRAME64 frame;
|
||||||
|
ZeroMemory(&frame, sizeof(STACKFRAME64));
|
||||||
|
DWORD machine_type;
|
||||||
|
#if defined(_M_IX86) || defined(__i386__)
|
||||||
|
machine_type = IMAGE_FILE_MACHINE_I386;
|
||||||
|
frame.AddrPC.Offset = context.Eip;
|
||||||
|
frame.AddrPC.Mode = AddrModeFlat;
|
||||||
|
frame.AddrFrame.Offset = context.Ebp;
|
||||||
|
frame.AddrFrame.Mode = AddrModeFlat;
|
||||||
|
frame.AddrStack.Offset = context.Esp;
|
||||||
|
frame.AddrStack.Mode = AddrModeFlat;
|
||||||
|
#elif defined(_M_X64) || defined(__x86_64__)
|
||||||
|
machine_type = IMAGE_FILE_MACHINE_AMD64;
|
||||||
|
frame.AddrPC.Offset = context.Rip;
|
||||||
|
frame.AddrPC.Mode = AddrModeFlat;
|
||||||
|
frame.AddrFrame.Offset = context.Rsp;
|
||||||
|
frame.AddrFrame.Mode = AddrModeFlat;
|
||||||
|
frame.AddrStack.Offset = context.Rsp;
|
||||||
|
frame.AddrStack.Mode = AddrModeFlat;
|
||||||
|
#elif defined(_M_IA64) || defined(__aarch64__)
|
||||||
|
machine_type = IMAGE_FILE_MACHINE_IA64;
|
||||||
|
frame.AddrPC.Offset = context.StIIP;
|
||||||
|
frame.AddrPC.Mode = AddrModeFlat;
|
||||||
|
frame.AddrFrame.Offset = context.IntSp;
|
||||||
|
frame.AddrFrame.Mode = AddrModeFlat;
|
||||||
|
frame.AddrBStore.Offset= context.RsBSP;
|
||||||
|
frame.AddrBStore.Mode = AddrModeFlat;
|
||||||
|
frame.AddrStack.Offset = context.IntSp;
|
||||||
|
frame.AddrStack.Mode = AddrModeFlat;
|
||||||
|
#else
|
||||||
|
#error "Cpptrace: StackWalk64 not supported for this platform yet"
|
||||||
|
#endif
|
||||||
|
ZeroMemory(&context, sizeof(CONTEXT));
|
||||||
|
StackWalk64(
|
||||||
|
machine_type,
|
||||||
|
proc,
|
||||||
|
thread,
|
||||||
|
&frame,
|
||||||
|
machine_type == IMAGE_FILE_MACHINE_I386 ? NULL : &context,
|
||||||
|
NULL,
|
||||||
|
SymFunctionTableAccess64,
|
||||||
|
SymGetModuleBase64,
|
||||||
|
NULL
|
||||||
|
);
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue
Block a user