radare2/libr/bin/dbginfo.c
HoundThe 914624a921
DWARF .debug_info parsing fix and improvement (#17007)
* Get .debug_info working for DWARF 2, 3
* Fill up missing information in abbrev string table
* Add 64bit dwarf option into debug_info
* Add DWARF4 and DWARF5 forms
* Add unit tests
* added basic test

Co-authored-by: Riccardo Schirone <sirmy15@gmail.com>
2020-06-19 19:25:59 +08:00

121 lines
3.0 KiB
C

/* radare - LGPL - Copyright 2009-2020 - nibble, pancake */
#include <r_types.h>
#include <r_bin.h>
R_API bool r_bin_addr2line(RBin *bin, ut64 addr, char *file, int len, int *line) {
r_return_val_if_fail (bin, false);
RBinFile *binfile = r_bin_cur (bin);
RBinObject *o = r_bin_cur_object (bin);
RBinPlugin *cp = r_bin_file_cur_plugin (binfile);
ut64 baddr = r_bin_get_baddr (bin);
if (cp && cp->dbginfo) {
if (o && addr >= baddr && addr < baddr + bin->cur->o->size) {
if (cp->dbginfo->get_line) {
return cp->dbginfo->get_line (
bin->cur, addr, file, len, line);
}
}
}
return false;
}
R_API char *r_bin_addr2text(RBin *bin, ut64 addr, int origin) {
r_return_val_if_fail (bin, NULL);
char file[4096];
int line;
char *out = NULL, *out2 = NULL;
char *file_nopath = NULL;
if (!bin->cur) {
return NULL;
}
char *key = r_str_newf ("0x%"PFMT64x, addr);
char *file_line = sdb_get (bin->cur->sdb_addrinfo, key, 0);
if (file_line) {
char *token = strchr (file_line, '|');
if (token) {
*token++ = 0;
line = atoi (token);
out = r_file_slurp_line (file_line, line, 0);
*token++ = ':';
} else {
return file_line;
}
}
free (key);
if (out) {
if (origin > 1) {
file_nopath = file_line;
} else {
file_nopath = strrchr (file_line, '/');
if (file_nopath) {
file_nopath++;
} else {
file_nopath = file_line;
}
}
if (origin) {
char *res = r_str_newf ("%s:%d%s%s",
file_nopath? file_nopath: "",
line, file_nopath? " ": "",
out? out: "");
free (out);
out = res;
}
free (file_line);
return out;
}
R_FREE (file_line);
file[0] = 0;
if (r_bin_addr2line (bin, addr, file, sizeof (file), &line)) {
if (bin->srcdir && *bin->srcdir) {
char *slash = strrchr (file, '/');
char *nf = r_str_newf ("%s/%s", bin->srcdir, slash? slash + 1: file);
strncpy (file, nf, sizeof (file) - 1);
free (nf);
}
// TODO: this is slow. must use a cached pool of mapped files and line:off entries
out = r_file_slurp_line (file, line, 0);
if (!out) {
if (origin > 1) {
file_nopath = file;
} else {
file_nopath = strrchr (file, '/');
if (file_nopath) {
file_nopath++;
} else {
file_nopath = file;
}
}
return r_str_newf ("%s:%d", file_nopath? file_nopath: "", line);
}
out2 = malloc ((strlen (file) + 64 + strlen (out)) * sizeof (char));
if (origin > 1) {
file_nopath = NULL;
} else {
file_nopath = strrchr (file, '/');
}
if (origin) {
snprintf (out2, strlen (file) + 63 + strlen (out), "%s:%d%s%s",
file_nopath? file_nopath + 1: file, line, *out? " ": "", out);
} else {
snprintf (out2, 64, "%s", out);
}
free (out);
}
return out2;
}
R_API char *r_bin_addr2fileline(RBin *bin, ut64 addr) {
r_return_val_if_fail (bin, NULL);
char file[1024];
int line = 0;
if (r_bin_addr2line (bin, addr, file, sizeof (file) - 1, &line)) {
char *file_nopath = strrchr (file, '/');
return r_str_newf ("%s:%d", file_nopath? file_nopath + 1: file, line);
}
return NULL;
}