Cthulhu  0.2.10
Cthulhu compiler collection
dbghelp.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
3 #include "backtrace/backtrace.h"
4 
5 #include "core/macros.h"
6 #include "core/win32.h" // IWYU pragma: keep
7 #include <dbghelp.h>
8 
9 #define MAX_NAME_SIZE 0x1000
10 
11 union disp_t {
12  DWORD disp;
13  DWORD64 disp64;
14 };
15 
17 const char *bt_backend(void)
18 {
19  return "dbghelp";
20 }
21 
22 static BOOL walk_stack(STACKFRAME *frame, CONTEXT *ctx, HANDLE process, HANDLE thread)
23 {
24  return StackWalk(
25  /* MachineType = */ IMAGE_FILE_MACHINE_AMD64,
26  /* hProcess = */ process,
27  /* kThread = */ thread,
28  /* StackFrame = */ frame,
29  /* Context = */ ctx,
30  /* ReadMemoryRoutine = */ NULL,
31  /* FunctionTableAccessRoutine = */ SymFunctionTableAccess64,
32  /* GetModuleBaseRoutine = */ SymGetModuleBase64,
33  /* TranslateAddress = */ NULL
34  );
35 }
36 
37 // split this out as a function so we can use it in the exception handler
38 // exception handlers provide a CONTEXT we can use to walk the stack
39 static void read_context_stack(CONTEXT *ctx, bt_trace_t callback, void *user)
40 {
41  HANDLE thread = GetCurrentThread();
42  HANDLE process = GetCurrentProcess();
43 
44  STACKFRAME stackframe = {
45  .AddrPC = {
46  .Offset = ctx->Rip,
47  .Mode = AddrModeFlat
48  },
49  .AddrFrame = {
50  .Offset = ctx->Rbp,
51  .Mode = AddrModeFlat
52  },
53  .AddrStack = {
54  .Offset = ctx->Rsp,
55  .Mode = AddrModeFlat
56  }
57  };
58 
59  while (walk_stack(&stackframe, ctx, process, thread))
60  {
61  bt_address_t frame = stackframe.AddrPC.Offset;
62 
63  callback(frame, user);
64  }
65 }
66 
67 void bt_read_inner(bt_trace_t callback, void *user)
68 {
69  CONTEXT ctx = { 0 };
70  RtlCaptureContext(&ctx);
71 
72  read_context_stack(&ctx, callback, user);
73 }
74 
76 {
77  union disp_t disp = { 0 };
78  IMAGEHLP_LINE64 line = { 0 };
79  HANDLE process = GetCurrentProcess();
80  text_t name = symbol->name;
81  text_t path = symbol->path;
82 
83  // cap the name size to a known maximum so we can put it on the stack rather than call malloc
84  // we do this because this may be called in a signal handler and we don't want to allocate
85  ULONG name_size = CT_MIN((ULONG)name.length, MAX_NAME_SIZE);
86  char buffer[sizeof(SYMBOL_INFO) + (MAX_NAME_SIZE - 1) * sizeof(TCHAR)];
87  memset(buffer, 0, sizeof(SYMBOL_INFO)); // only zero the symbol info struct
88 
89  PSYMBOL_INFO info = (PSYMBOL_INFO)buffer;
90  info->SizeOfStruct = sizeof(SYMBOL_INFO);
91  info->MaxNameLen = name_size;
92 
93  bt_resolve_t resolve = eResolveNothing;
94 
95  if (SymFromAddr(process, frame, &disp.disp64, info))
96  {
97  resolve |= eResolveName;
98 
99  if (SymGetLineFromAddr64(process, frame, &disp.disp, &line))
100  {
101  // subtract 1 from the line number because dbghelp is 1-indexed
102  symbol->line = line.LineNumber - 1;
103  strcpy_s(path.text, path.length, line.FileName);
104 
105  resolve |= eResolveLine | eResolveFile;
106  }
107 
108  // even though we're a C codebase still run the demangler to catch C++ code
109  // that people consuming this library might be using
110  if (UnDecorateSymbolName(info->Name, name.text, name_size, UNDNAME_COMPLETE))
111  {
112  resolve |= eResolveDemangledName;
113  }
114  else
115  {
116  // copy the mangled name if we can't demangle it
117  strcpy_s(name.text, name_size, info->Name);
118  }
119  }
120 
121  return resolve;
122 }
123 
124 static LONG WINAPI bt_exception_handler(EXCEPTION_POINTERS *exception)
125 {
126  /* ignore C++ exceptions */
127  if (exception->ExceptionRecord->ExceptionCode == 0xe06d7363)
128  return EXCEPTION_CONTINUE_SEARCH;
129 
130  gSystemError.begin(exception->ExceptionRecord->ExceptionCode, gSystemError.user);
131 
132  read_context_stack(exception->ContextRecord, gSystemError.next, gSystemError.user);
133 
135 
136  return EXCEPTION_EXECUTE_HANDLER;
137 }
138 
139 void bt_init(void)
140 {
141  SymInitialize(GetCurrentProcess(), NULL, TRUE);
142  SymSetOptions(SYMOPT_LOAD_LINES);
143 
144  SetUnhandledExceptionFilter(bt_exception_handler);
145 }
146 
147 void bt_update(void)
148 {
149  SymRefreshModuleList(GetCurrentProcess());
150 }
bt_resolve_t bt_resolve_inner(bt_address_t frame, bt_symbol_t *symbol)
Definition: dbghelp.c:75
#define MAX_NAME_SIZE
Definition: dbghelp.c:9
void bt_read_inner(bt_trace_t callback, void *user)
Definition: dbghelp.c:67
#define STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
void bt_init(void)
initialize the backtrace backend
Definition: dbghelp.c:139
uint_least64_t bt_address_t
an address of a symbol
Definition: backtrace.h:26
void(* bt_trace_t)(bt_address_t frame, void *user)
user callback for bt_read
Definition: backtrace.h:74
bt_resolve_t
how much of a frame was reconstructed
Definition: backtrace.h:47
void bt_update(void)
update the loaded module cache
Definition: dbghelp.c:147
STA_DECL const char * bt_backend(void)
get the backtrace backend name
Definition: dbghelp.c:17
CT_BACKTRACE_API bt_error_t gSystemError
the global system error handler
Definition: common.c:11
@ eResolveDemangledName
the symbol name was demangled
Definition: backtrace.h:60
@ eResolveLine
the line number was found
Definition: backtrace.h:53
@ eResolveNothing
nothing was resolved
Definition: backtrace.h:49
@ eResolveName
the symbol name was found
Definition: backtrace.h:57
@ eResolveFile
the file path was found
Definition: backtrace.h:64
#define CT_MIN(L, R)
Definition: macros.h:38
bt_trace_t next
called once for each frame
Definition: backtrace.h:97
bt_error_begin_t begin
called once when a system error occurs
Definition: backtrace.h:91
void * user
user data to pass to the callbacks
Definition: backtrace.h:100
bt_error_end_t end
called after all frames have been collected
Definition: backtrace.h:94
a symbol
Definition: backtrace.h:33
text_t name
a buffer to hold the name
Definition: backtrace.h:39
text_t path
a buffer to hold the path to the file
Definition: backtrace.h:42
source_line_t line
the line number
Definition: backtrace.h:36
a range of text
Definition: text.h:14
size_t length
the number of characters in the text
Definition: text.h:19
Definition: dbghelp.c:11
DWORD64 disp64
Definition: dbghelp.c:13
DWORD disp
Definition: dbghelp.c:12