1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889 |
- #include <libgcc_s.h>
- #include <execinfo.h>
- #include <dlfcn.h>
- #include <stdlib.h>
- #include <unwind.h>
- #include <assert.h>
- #include <stdio.h>
- struct trace_arg
- {
- void **array;
- int cnt, size;
- };
- #ifdef SHARED
- static _Unwind_Reason_Code (*unwind_backtrace) (_Unwind_Trace_Fn, void *);
- static _Unwind_Ptr (*unwind_getip) (struct _Unwind_Context *);
- static void backtrace_init (void)
- {
- void *handle = dlopen (LIBGCC_S_SO, RTLD_LAZY);
- if (handle == NULL
- || ((unwind_backtrace = dlsym (handle, "_Unwind_Backtrace")) == NULL)
- || ((unwind_getip = dlsym (handle, "_Unwind_GetIP")) == NULL)) {
- printf(LIBGCC_S_SO " must be installed for backtrace to work\n");
- abort();
- }
- }
- #else
- # define unwind_backtrace _Unwind_Backtrace
- # define unwind_getip _Unwind_GetIP
- #endif
- static _Unwind_Reason_Code
- backtrace_helper (struct _Unwind_Context *ctx, void *a)
- {
- struct trace_arg *arg = a;
- assert (unwind_getip != NULL);
-
- if (arg->cnt != -1)
- arg->array[arg->cnt] = (void *) unwind_getip (ctx);
- if (++arg->cnt == arg->size)
- return _URC_END_OF_STACK;
- return _URC_NO_REASON;
- }
- int backtrace (void **array, int size)
- {
- struct trace_arg arg = { .array = array, .size = size, .cnt = -1 };
- #ifdef SHARED
- if (unwind_backtrace == NULL)
- backtrace_init();
- #endif
- if (size >= 1)
- unwind_backtrace (backtrace_helper, &arg);
- return arg.cnt != -1 ? arg.cnt : 0;
- }
|