mirror of
https://github.com/tmate-io/tmate.git
synced 2024-11-29 11:33:29 +01:00
3f6c6d4447
backtrace(3) and execinfo.h are GNU extensions and may or may not be available, and they may be provided via libexecinfo. Fix detection of libexecinfo and allow building without any support of backtrace, in which case we let kernel create core dump. Fixes #116 #117
101 lines
1.8 KiB
C
101 lines
1.8 KiB
C
#ifdef HAVE_EXECINFO_H
|
|
#include <execinfo.h>
|
|
#endif
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <regex.h>
|
|
#include <signal.h>
|
|
#include "tmate.h"
|
|
|
|
#ifndef HAVE_BACKTRACE
|
|
|
|
void tmate_print_stack_trace(void) {}
|
|
void tmate_catch_sigsegv(void) {}
|
|
|
|
#else
|
|
|
|
#if DEBUG
|
|
|
|
static int print_resolved_stack_frame(const char *frame)
|
|
{
|
|
char file[100];
|
|
char cmd[200];
|
|
char output[300];
|
|
char address[20];
|
|
char *line;
|
|
FILE *ps;
|
|
|
|
static regex_t _regex;
|
|
static regex_t *regex;
|
|
regmatch_t matches[3];
|
|
|
|
if (!regex) {
|
|
if (regcomp(&_regex, "(.+)\\(\\) \\[([^]]+)\\]", REG_EXTENDED))
|
|
return -1;
|
|
regex = &_regex;
|
|
}
|
|
|
|
if (regexec(regex, frame, 3, matches, 0))
|
|
return -1;
|
|
|
|
memcpy(file, &frame[matches[1].rm_so], matches[1].rm_eo - matches[1].rm_so);
|
|
file[matches[1].rm_eo - matches[1].rm_so] = 0;
|
|
|
|
memcpy(address, &frame[matches[2].rm_so], matches[2].rm_eo - matches[2].rm_so);
|
|
address[matches[2].rm_eo - matches[2].rm_so] = 0;
|
|
|
|
sprintf(cmd, "addr2line -e %s %s -f -p -s", file, address);
|
|
|
|
ps = popen(cmd, "r");
|
|
if (!ps)
|
|
return -1;
|
|
|
|
line = fgets(output, sizeof(output), ps);
|
|
pclose(ps);
|
|
|
|
if (!line)
|
|
return -1;
|
|
|
|
line[strlen(line)-1] = 0; /* remove \n */
|
|
tmate_debug("%s(%s) [%s]", file, line, address);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
void tmate_print_stack_trace(void)
|
|
{
|
|
void *array[20];
|
|
size_t size;
|
|
char **strings;
|
|
size_t i;
|
|
|
|
size = backtrace (array, 20);
|
|
strings = backtrace_symbols (array, size);
|
|
|
|
tmate_info ("============ %zd stack frames ============", size);
|
|
|
|
for (i = 1; i < size; i++) {
|
|
#if DEBUG
|
|
if (print_resolved_stack_frame(strings[i]) < 0)
|
|
#endif
|
|
tmate_info("%s", strings[i]);
|
|
}
|
|
|
|
free (strings);
|
|
}
|
|
|
|
|
|
static void handle_sigsegv(__unused int sig)
|
|
{
|
|
/* TODO send stack trace to server */
|
|
tmate_info("CRASH, printing stack trace");
|
|
tmate_print_stack_trace();
|
|
tmate_fatal("CRASHED");
|
|
}
|
|
|
|
void tmate_catch_sigsegv(void)
|
|
{
|
|
signal(SIGSEGV, handle_sigsegv);
|
|
}
|
|
#endif
|