efl/src/bin/eina/eina_btlog.c

199 lines
5.4 KiB
C

#include <Eina.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
// right now this is quick and dirty and may have some parsing ... frailty,
// so don't put malicious data through it... :) but cat in eina bt's through
// this to get a nicely clean and readable bt with filenames of binaries,
// shared objects, source files, and line numbers. even nicely colored and
// columnated. this is more the start of a bunch of debug tools for efl to make
// it easier to identify issues.
//
// how to use:
//
// cat mybacktrace.txt | eina_btlog
//
// (or just run it and copy & paste in on stdin - what i do mostly, and out
// pops a nice backtrace, hit ctrl+d to end)
typedef struct _Bt Bt;
struct _Bt
{
char *bin_dir;
char *bin_name;
char *file_dir;
char *file_name;
char *func_name;
int line;
};
static void
path_split(const char *path, char **dir, char **file)
{
const char *p = strrchr(path, '/');
if (!path)
{
*dir = NULL;
*file = NULL;
return;
}
if (!p)
{
*dir = NULL;
*file = strdup(path);
return;
}
*dir = malloc(p - path + 1);
if (!dir)
{
*dir = NULL;
*file = NULL;
return;
}
strncpy(*dir, path, p - path);
(*dir)[p - path] = 0;
*file = strdup(p + 1);
}
static Eina_Bool
_addr2line(const char *bin_dir, const char *bin_name, unsigned long long addr,
char **file_dir, char **file_name, char **func_name, int *file_line)
{
char buf[4096], func[4096], *f1 = NULL, *f2 = NULL;
Eina_Bool ok = EINA_FALSE;
int line;
FILE *p;
snprintf(buf, sizeof(buf), "addr2line -f -e %s/%s -C -a 0x%llx",
bin_dir, bin_name, addr);
p = popen(buf, "r");
if (!p) return EINA_FALSE;
fscanf(p, "%s\n", buf);
if (fscanf(p, "%s\n", func) == 1)
{
if (fscanf(p, "%[^:]:%i\n", buf, &line) == 2)
{
path_split(buf, &(f1), &(f2));
if ((!f1) || (!f2))
{
free(f1);
free(f2);
pclose(p);
return EINA_FALSE;
}
}
else
{
f1 = strdup("??");
f2 = strdup("??");
}
*file_dir = f1;
*file_name = f2;
*func_name = strdup(func);
*file_line = line;
ok = EINA_TRUE;
}
pclose(p);
return ok;
}
static Eina_List *
bt_append(Eina_List *btl, const char *btline)
{
Bt *bt = calloc(1, sizeof(Bt));
if (!bt) return btl;
char *bin = strdup(btline);
unsigned long long offset = 0, base = 0;
// parse:
// /usr/local/lib/libeina.so.1 0x1ec88
// /usr/local/lib/libelementary.so.1 0x10f695
// /usr/local/lib/libeo.so.1 0xa474
// /usr/local/lib/libelementary.so.1 0x139bd6
// /usr/local/bin/elementary_test 0x8196d
// /usr/local/bin/elementary_test 0x81b6a
if (sscanf(btline, "%s %llx %llx", bin, &offset, &base) == 3)
{
path_split(bin, &(bt->bin_dir), &(bt->bin_name));
if (!bt->bin_dir) bt->bin_dir = strdup("");
if (!bt->bin_name) bt->bin_name = strdup("");
if (!_addr2line(bt->bin_dir, bt->bin_name, offset - base,
&(bt->file_dir), &(bt->file_name),
&(bt->func_name), &(bt->line)))
{
if (!_addr2line(bt->bin_dir, bt->bin_name, offset,
&(bt->file_dir), &(bt->file_name),
&(bt->func_name), &(bt->line)))
{
bt->file_dir = strdup("");
bt->file_name = strdup("");
bt->func_name = strdup("");
}
}
btl = eina_list_append(btl, bt);
}
free(bin);
return btl;
}
int
main(int argc, char **argv)
{
Eina_List *btl = NULL, *l;
char buf[4096];
Bt *bt;
int cols[6] = { 0 }, len, i;
eina_init();
while (fgets(buf, sizeof(buf) - 1, stdin))
{
btl = bt_append(btl, buf);
}
EINA_LIST_FOREACH(btl, l, bt)
{
len = strlen(bt->bin_dir);
if (len > cols[0]) cols[0] = len;
len = strlen(bt->bin_name);
if (len > cols[1]) cols[1] = len;
len = strlen(bt->file_dir);
if (len > cols[2]) cols[2] = len;
len = strlen(bt->file_name);
if (len > cols[3]) cols[3] = len;
snprintf(buf, sizeof(buf), "%i", bt->line);
len = strlen(buf);
if (len > cols[4]) cols[4] = len;
len = strlen(bt->func_name);
if (len > cols[5]) cols[5] = len;
}
EINA_LIST_FOREACH(btl, l, bt)
{
len = strlen(bt->bin_dir);
for (i = 0; i < (cols[0] - len); i++) printf(" ");
printf("\033[34m%s\033[01m\033[36m/\033[37m%s\033[0m",
bt->bin_dir, bt->bin_name);
len = strlen(bt->bin_name);
for (i = 0; i < (cols[1] - len); i++) printf(" ");
printf(" | ");
len = strlen(bt->file_dir);
for (i = 0; i < (cols[2] - len); i++) printf(" ");
printf("\033[34m%s\033[01m\033[36m/\033[37m%s\033[0m",
bt->file_dir, bt->file_name);
len = strlen(bt->file_name);
for (i = 0; i < (cols[3] - len); i++) printf(" ");
printf(" : ");
snprintf(buf, sizeof(buf), "%i", bt->line);
len = strlen(buf);
for (i = 0; i < (cols[4] - len); i++) printf(" ");
printf("\033[01m\033[33m%s\033[0m @ \033[32m%s\033[36m()", buf, bt->func_name);
printf("\033[0m\n");
}
return 0;
}