nwaku/vendor/nim-libbacktrace/libbacktrace_wrapper.c

438 lines
12 KiB
C

/*
* Copyright (c) 2019-2021 Status Research & Development GmbH
* Licensed under either of
* * Apache License, version 2.0,
* * MIT license
* at your option.
* This file may not be copied, modified, or distributed except according to
* those terms.
*/
#include <backtrace-supported.h>
#include <backtrace.h>
#include <errno.h>
#include <inttypes.h>
#include <libgen.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "libbacktrace_wrapper.h"
// https://stackoverflow.com/a/44383330
#ifdef _WIN32
# ifdef _WIN64
# define PRI_SIZET PRIu64
# else
# define PRI_SIZET PRIu32
# endif
#else
# define PRI_SIZET "zu"
#endif
// https://sourceforge.net/p/mingw/mailman/mingw-users/thread/46C99879.8070205@cox.net/
#ifdef __MINGW32__
# define snprintf __mingw_snprintf
#endif
// macOS Clang wants this before the WAI_MALLOC define
static void *xmalloc(size_t size)
{
void *res = malloc(size);
if (res == NULL) {
fprintf(stderr, "FATAL: malloc() failed to allocate %" PRI_SIZET " bytes.\n", size);
exit(1);
}
return res;
}
static void xfree_inner(void **ptr)
{
if (ptr == NULL) {
fprintf(stderr, "FATAL: xfree_inner() was called with a NULL pointer.\n");
exit(1);
} else {
free(*ptr);
*ptr = NULL;
}
}
#define xfree(ptr) xfree_inner((void**) &ptr)
#define WAI_MALLOC(size) xmalloc(size)
#include "vendor/whereami/src/whereami.h"
// Yes, this is ugly. Using the Nim compiler as a build system is uglier.
#include "vendor/whereami/src/whereami.c"
// Saw this limit somewhere in the Nim compiler source.
#define MAX_BACKTRACE_LINES 128
#define INITIAL_LINE_SIZE 100
#define DEBUG_ENV_VAR_NAME "NIM_LIBBACKTRACE_DEBUG"
static __thread int debug = 0;
struct callback_data {
struct debugging_info *di_data;
int next_index;
int max_length;
int nim_main_module_seen; // Did we already see NimMainModule?
};
struct simple_callback_data {
uintptr_t *program_counters;
int next_index;
int max_length;
};
static __thread struct backtrace_state *state;
// Is this going to be zero in all threads?
static __thread int cb_data_initialised = 0;
static char *xstrdup(const char *s)
{
char *res = strdup(s);
if (res == NULL) {
fprintf(stderr, "FATAL: strdup() failure.\n");
exit(1);
}
return res;
}
static void error_callback(void *data __attribute__ ((__unused__)),
const char *msg, int errnum)
{
fprintf(stderr, "libbacktrace error: %s (%d)\n", msg, errnum);
}
static int strings_equal(const char *str1, const char *str2)
{
if (!str1 || !str2) {
return 0;
} else {
size_t len2 = strlen(str2);
return strlen(str1) == len2 && strncmp(str1, str2, len2) == 0;
}
}
static int string_starts_with(const char *str1, const char *str2)
{
if (!str1 || !str2) {
return 0;
} else {
size_t len2 = strlen(str2);
return strlen(str1) >= len2 && strncmp(str1, str2, len2) == 0;
}
}
#ifdef __cplusplus
# include <cxxabi.h>
#endif // __cplusplus
static char *demangle(const char *function)
{
if (function == NULL) {
fprintf(stderr, "demangle() called with a NULL pointer. Aborting.\n");
exit(1);
}
char *res = xstrdup(function);
#ifdef __cplusplus
// C++ function name demangling.
size_t demangled_len;
int status;
char* demangled = abi::__cxa_demangle(function, NULL, &demangled_len, &status);
if (demangled && status == 0) {
demangled[demangled_len] = '\0';
// Get rid of function parenthesis and params.
char *par_pos = strchr(demangled, '(');
if (par_pos)
*par_pos = '\0';
xfree(res);
res = demangled;
}
#endif // __cplusplus
// Nim demangling.
char *pos = strstr(res, "__");
if (pos)
*pos = '\0';
return res;
}
static int success_callback(void *data, uintptr_t pc __attribute__((unused)),
const char *filename, int lineno, const char *function)
{
// clang++ makes us do all these pointer casts.
struct callback_data *cb_data = (struct callback_data*) data;
if (cb_data->next_index >= cb_data->max_length)
return 1; // Stop building the backtrace.
if (function == NULL || filename == NULL) {
// see https://github.com/status-im/nim-libbacktrace/issues/9, we need to keep going here.
return 0;
}
char *demangled_function = demangle(function);
// skip internal Nim functions
if ((strings_equal(demangled_function, "NimMainInner") ||
strings_equal(demangled_function, "NimMain") ||
strings_equal(demangled_function, "main")) &&
cb_data->nim_main_module_seen) {
/*
* If we skip them unconditionally, we may end up with an empty
* backtrace when `-d:release` leads to NimMainModule being
* inlined.
*/
if (!debug) {
xfree(demangled_function);
return 1; // Stop building the backtrace.
}
}
// these ones appear when we're used inside the Nim compiler
if (strings_equal(demangled_function, "auxWriteStackTraceWithOverride") ||
strings_equal(demangled_function, "rawWriteStackTrace") ||
strings_equal(demangled_function, "writeStackTrace") ||
strings_equal(demangled_function, "raiseExceptionAux") ||
string_starts_with(demangled_function, "raiseExceptionEx")) {
if (!debug) {
xfree(demangled_function);
return 0; // Skip it, but continue the backtrace.
}
}
// Replace "NimMainModule" with the file name (minus the extension).
if (strings_equal(demangled_function, "NimMainModule")) {
cb_data->nim_main_module_seen = 1;
// "/foo/bar/test2.nim" -> "test2"
char *nim_file = xstrdup(filename);
char *pos = basename(nim_file);
size_t len = strlen(pos);
if (len > 4)
pos[len - 4] = '\0';
xfree(demangled_function);
demangled_function = xstrdup(pos);
xfree(nim_file);
}
cb_data->di_data[cb_data->next_index].filename = xstrdup(filename);
cb_data->di_data[cb_data->next_index].lineno = lineno;
cb_data->di_data[cb_data->next_index].function = demangled_function;
cb_data->next_index++;
return 0;
}
static int simple_success_callback(void *data, uintptr_t pc)
{
struct simple_callback_data *scb_data = (struct simple_callback_data*)data;
if (scb_data->next_index >= scb_data->max_length) {
return 1; // stop traversing the stack
} else {
scb_data->program_counters[scb_data->next_index] = pc;
scb_data->next_index++;
return 0; // continue traversing the stack
}
}
static char *internal_init(void)
{
if (!cb_data_initialised) {
cb_data_initialised = 1;
char *debug_env_var_value = getenv(DEBUG_ENV_VAR_NAME);
if (strings_equal(debug_env_var_value, "1"))
debug = 1;
// Using https://github.com/gpakosz/whereami
int self_exec_path_length = wai_getExecutablePath(NULL, 0, NULL);
if (self_exec_path_length == -1)
return xstrdup("whereami error: could not get the program's path on this platform.\n");
char *self_exec_path = (char*) xmalloc(self_exec_path_length + 1);
wai_getExecutablePath(self_exec_path, self_exec_path_length, NULL);
self_exec_path[self_exec_path_length] = '\0';
/*
* We shouldn't initialise this state more than once per thread:
* https://github.com/ianlancetaylor/libbacktrace/issues/13
*/
state = backtrace_create_state(self_exec_path, 0, error_callback, NULL);
}
return xstrdup("");
}
// The returned array needs to be freed by the caller.
uintptr_t *get_program_counters_c(int max_length, int skip)
{
// we use a sentinel value (0) to mark the end
size_t program_counters_size = sizeof(uintptr_t) * (max_length + 1);
uintptr_t *program_counters = (uintptr_t*) xmalloc(program_counters_size);
memset(program_counters, '\0', program_counters_size);
#ifdef BACKTRACE_SUPPORTED
struct simple_callback_data scb_data = {program_counters, 0, max_length};
char *err = internal_init();
if (!strings_equal(err , "")) {
error_callback(NULL, err, 0);
xfree(err);
return program_counters;
}
xfree(err);
// Get the program counters.
if (state != NULL)
backtrace_simple(state, skip, simple_success_callback, error_callback, &scb_data);
#endif // BACKTRACE_SUPPORTED
return program_counters;
}
// The returned array needs to be freed by the caller.
struct debugging_info *get_debugging_info_c(uintptr_t *program_counters, int max_length)
{
// we use a sentinel value to mark the end
size_t di_data_size = sizeof(struct debugging_info) * (max_length + 1);
struct debugging_info *di_data = (struct debugging_info*) xmalloc(di_data_size);
memset(di_data, '\0', di_data_size);
#ifdef BACKTRACE_SUPPORTED
struct callback_data cb_data;
memset(&cb_data, '\0', sizeof(struct callback_data));
cb_data.di_data = di_data;
cb_data.max_length = max_length;
char *err = internal_init();
if (!strings_equal(err , "")) {
xfree(err);
return di_data;
}
xfree(err);
int i = 0;
int res = 0;
while (program_counters[i] != 0) {
/*
* "success_callback()" may be called multiple times for the
* same program counter, if inlined functions are involved.
*/
res = backtrace_pcinfo(state, program_counters[i], success_callback, error_callback, &cb_data);
// We stop when the callback decided to skip something.
if (res != 0)
break;
i++;
}
#endif // BACKTRACE_SUPPORTED
return di_data;
}
// The returned string needs to be freed by the caller.
char *get_backtrace_max_length_c(int max_length, int skip)
{
#ifdef BACKTRACE_SUPPORTED
char **backtrace_lines = (char**) xmalloc(sizeof(char*) * max_length);
int *backtrace_line_lengths = (int*) xmalloc(sizeof(int) * max_length);
int last_line_index;
char *err = internal_init();
if (!strings_equal(err , ""))
return err;
xfree(err);
if (state != NULL) {
// Get the program counters.
int skip_functions = 0;
if (!debug)
skip_functions = skip;
uintptr_t *program_counters = get_program_counters_c(max_length, skip_functions);
/*
* Get the filename, line number and function name for each
* program counter. In the case of inlined functions, we may
* get multiple hits from DWARF metadata for the same program
* counter. That's OK, we want those.
*/
struct debugging_info *di_data = get_debugging_info_c(program_counters, max_length);
xfree(program_counters);
// String building.
int backtrace_line_size = INITIAL_LINE_SIZE;
char *backtrace_line;
int output_size; // Excludes the terminating null byte.
int di_data_index = 0;
while (di_data[di_data_index].filename != NULL) {
backtrace_line = (char*) xmalloc(backtrace_line_size);
while (1) {
// We're mirroring Nim's default stack trace format.
output_size = snprintf(backtrace_line, backtrace_line_size,
"%s(%d) %s\n",
di_data[di_data_index].filename,
di_data[di_data_index].lineno,
di_data[di_data_index].function);
if (output_size + 1 <= backtrace_line_size) {
break;
} else {
backtrace_line_size *= 2;
xfree(backtrace_line);
backtrace_line = (char*) xmalloc(backtrace_line_size);
}
}
backtrace_lines[di_data_index] = backtrace_line;
backtrace_line_lengths[di_data_index] = output_size;
di_data_index++;
}
last_line_index = di_data_index - 1;
} else {
return xstrdup(""); // The error callback has already been called.
}
int total_length = 0;
int i;
// The Nim tradition wants them in reverse order.
for (i = last_line_index; i >= 0; i--) {
if (backtrace_lines[i] != NULL)
total_length += backtrace_line_lengths[i];
}
char *backtrace = (char*) xmalloc(total_length + 1);
char *last_null_byte = backtrace;
*last_null_byte = '\0';
// Produce the string result.
for (i = last_line_index; i >= 0; i--) {
last_null_byte = (char*)memccpy(last_null_byte,
backtrace_lines[i],
'\0',
backtrace_line_lengths[i] + 1) - 1;
xfree(backtrace_lines[i]);
}
// Cleanup.
xfree(backtrace_lines);
xfree(backtrace_line_lengths);
return backtrace;
#else // BACKTRACE_SUPPORTED
return xstrdup("ERROR: libbacktrace is not supported on this platform.\n");
#endif // BACKTRACE_SUPPORTED
}
// The returned string needs to be freed by the caller.
char *get_backtrace_c(void)
{
return get_backtrace_max_length_c(MAX_BACKTRACE_LINES, 3);
}