aboutsummaryrefslogtreecommitdiff
path: root/btrace.c
blob: e9ade89fa35e6fd8ff096315cb551b05713b3f73 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
#include <execinfo.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/wait.h>

#define STACKTRACES 10


static char *exec_path = NULL;


void bt_set_arg0(const char *arg0) {
	if (!exec_path && arg0) {
		exec_path = strdup(arg0);
	}
}

static void addr_to_array(char **strings, size_t siz, char **result) {
	char *start, *end;

	for (size_t i = 0; i < siz; ++i) {
		start = strchr(strings[i], '+');
		if (start) {
			end = strchr(start, ')');
			start++;
			if (end)
				result[i] = strndup(start, end-start);
			else
				result[i] = strdup("");
		}
	}
}

void bt_print_trace (void) {
	void *array[STACKTRACES];
	size_t bt_size;
	char **strings;
	size_t i;
	char **addrs;

	bt_size = backtrace(array, STACKTRACES);
	strings = backtrace_symbols(array, bt_size);

	printf("\n\n[1] Obtained %zd stack frames.\n", bt_size);
	for (i = 0; i < bt_size; ++i)
		printf("%s\n", strings[i]);

	/* addr2line -p -e ./btrace -f -i [addresses] */
	if (exec_path) {
		addrs = (char**) calloc(bt_size, sizeof(*addrs));

		addr_to_array(strings, bt_size, addrs);
		printf("\n\n[2] Run for more info: addr2line -p -e %s -f -i", exec_path);
		for (i = 0; i < bt_size; ++i) {
			printf(" %s", addrs[i]);
		}
		printf("%s\n", "");

		for (i = 0; i < bt_size; ++i)
			if (addrs[i])
				free(addrs[i]);
		free(addrs);
	}

	free(strings);
}

void bt_sighandler(int signum) {
	bt_print_trace();
	switch (signum) {
		case SIGTERM:
		case SIGABRT:
		case SIGSEGV:
			exit(1);
	}
}




/* A dummy function to make the backtrace more interesting. */
void dummy_function(void) {
	bt_print_trace();
}

#include <assert.h>
int main(int argc, char **argv) {
	bt_set_arg0(argv[0]);
	dummy_function();
	signal(SIGTERM, bt_sighandler);
	signal(SIGABRT, bt_sighandler);
	signal(SIGSEGV, bt_sighandler);

	//assert(0);
	*(int*)(NULL) = 0;
	return 0;
}