9 #define MAX_NAME_SIZE 0x1000
22 static BOOL walk_stack(STACKFRAME *frame, CONTEXT *ctx, HANDLE process, HANDLE thread)
25 IMAGE_FILE_MACHINE_AMD64,
31 SymFunctionTableAccess64,
39 static void read_context_stack(CONTEXT *ctx,
bt_trace_t callback,
void *user)
41 HANDLE thread = GetCurrentThread();
42 HANDLE process = GetCurrentProcess();
44 STACKFRAME stackframe = {
59 while (walk_stack(&stackframe, ctx, process, thread))
63 callback(frame, user);
70 RtlCaptureContext(&ctx);
72 read_context_stack(&ctx, callback, user);
78 IMAGEHLP_LINE64 line = { 0 };
79 HANDLE process = GetCurrentProcess();
86 char buffer[
sizeof(SYMBOL_INFO) + (
MAX_NAME_SIZE - 1) *
sizeof(TCHAR)];
87 memset(buffer, 0,
sizeof(SYMBOL_INFO));
89 PSYMBOL_INFO info = (PSYMBOL_INFO)buffer;
90 info->SizeOfStruct =
sizeof(SYMBOL_INFO);
91 info->MaxNameLen = name_size;
95 if (SymFromAddr(process, frame, &
disp.disp64, info))
99 if (SymGetLineFromAddr64(process, frame, &
disp.disp, &line))
102 symbol->
line = line.LineNumber - 1;
103 strcpy_s(path.text, path.
length, line.FileName);
110 if (UnDecorateSymbolName(info->Name, name.text, name_size, UNDNAME_COMPLETE))
117 strcpy_s(name.text, name_size, info->Name);
124 static LONG WINAPI bt_exception_handler(EXCEPTION_POINTERS *exception)
127 if (exception->ExceptionRecord->ExceptionCode == 0xe06d7363)
128 return EXCEPTION_CONTINUE_SEARCH;
136 return EXCEPTION_EXECUTE_HANDLER;
141 SymInitialize(GetCurrentProcess(), NULL, TRUE);
142 SymSetOptions(SYMOPT_LOAD_LINES);
144 SetUnhandledExceptionFilter(bt_exception_handler);
149 SymRefreshModuleList(GetCurrentProcess());
bt_resolve_t bt_resolve_inner(bt_address_t frame, bt_symbol_t *symbol)
void bt_read_inner(bt_trace_t callback, void *user)
#define STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
void bt_init(void)
initialize the backtrace backend
uint_least64_t bt_address_t
an address of a symbol
void(* bt_trace_t)(bt_address_t frame, void *user)
user callback for bt_read
bt_resolve_t
how much of a frame was reconstructed
void bt_update(void)
update the loaded module cache
STA_DECL const char * bt_backend(void)
get the backtrace backend name
CT_BACKTRACE_API bt_error_t gSystemError
the global system error handler
@ eResolveDemangledName
the symbol name was demangled
@ eResolveLine
the line number was found
@ eResolveNothing
nothing was resolved
@ eResolveName
the symbol name was found
@ eResolveFile
the file path was found
bt_trace_t next
called once for each frame
bt_error_begin_t begin
called once when a system error occurs
void * user
user data to pass to the callbacks
bt_error_end_t end
called after all frames have been collected
text_t name
a buffer to hold the name
text_t path
a buffer to hold the path to the file
source_line_t line
the line number
size_t length
the number of characters in the text