mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-24 14:54:54 +00:00
4554 lines
124 KiB
C
4554 lines
124 KiB
C
/* radare - LGPL - Copyright 2011-2022 - earada, pancake */
|
|
|
|
#include <r_core.h>
|
|
#include <r_config.h>
|
|
#include <r_util.h>
|
|
#include <r_util/r_time.h>
|
|
|
|
#define is_in_range(at, from, sz) ((at) >= (from) && (at) < ((from) + (sz)))
|
|
|
|
#define VA_FALSE 0
|
|
#define VA_TRUE 1
|
|
#define VA_NOREBASE 2
|
|
|
|
#define LOAD_BSS_MALLOC 0
|
|
|
|
#define IS_MODE_SET(mode) ((mode) & R_MODE_SET)
|
|
#define IS_MODE_SIMPLE(mode) ((mode) & R_MODE_SIMPLE)
|
|
#define IS_MODE_SIMPLEST(mode) ((mode) & R_MODE_SIMPLEST)
|
|
#define IS_MODE_JSON(mode) ((mode) & R_MODE_JSON)
|
|
#define IS_MODE_RAD(mode) ((mode) & R_MODE_RADARE)
|
|
#define IS_MODE_EQUAL(mode) ((mode) & R_MODE_EQUAL)
|
|
#define IS_MODE_NORMAL(mode) (!(mode))
|
|
#define IS_MODE_CLASSDUMP(mode) ((mode) & R_MODE_CLASSDUMP)
|
|
|
|
// dup from cmd_info
|
|
#define PAIR_WIDTH "9"
|
|
|
|
#define bprintf if (binfile && binfile->rbin && binfile->rbin->verbose) eprintf
|
|
|
|
static void pair(const char *key, const char *val) {
|
|
if (!val || !*val) {
|
|
return;
|
|
}
|
|
r_cons_printf ("%-"PAIR_WIDTH"s%s\n", key, val);
|
|
}
|
|
|
|
static void pair_bool(PJ *pj, const char *key, bool val) {
|
|
if (pj) {
|
|
pj_kb (pj, key, val);
|
|
} else {
|
|
pair (key, r_str_bool (val));
|
|
}
|
|
}
|
|
|
|
static void pair_int(PJ *pj, const char *key, int val) {
|
|
if (pj) {
|
|
pj_ki (pj, key, val);
|
|
} else {
|
|
char decnum[64];
|
|
snprintf (decnum, sizeof (decnum), "%d", val);
|
|
pair (key, decnum);
|
|
}
|
|
}
|
|
|
|
static void pair_ut64(PJ *pj, const char *key, ut64 val) {
|
|
if (pj) {
|
|
pj_kn (pj, key, val);
|
|
} else {
|
|
char decnum[64];
|
|
snprintf (decnum, sizeof (decnum), "%"PFMT64d, val);
|
|
pair (key, decnum);
|
|
}
|
|
}
|
|
|
|
static char *__filterQuotedShell(const char *arg) {
|
|
r_return_val_if_fail (arg, NULL);
|
|
char *a = malloc (strlen (arg) + 1);
|
|
if (!a) {
|
|
return NULL;
|
|
}
|
|
char *b = a;
|
|
while (*arg) {
|
|
switch (*arg) {
|
|
case ' ':
|
|
case '=':
|
|
case '"':
|
|
case '\\':
|
|
case '\r':
|
|
case '\n':
|
|
break;
|
|
default:
|
|
*b++ = *arg;
|
|
break;
|
|
}
|
|
arg++;
|
|
}
|
|
*b = 0;
|
|
return a;
|
|
}
|
|
// TODO: move into libr/util/name.c
|
|
static char *__filterShell(const char *arg) {
|
|
r_return_val_if_fail (arg, NULL);
|
|
char *a = malloc (strlen (arg) + 1);
|
|
if (!a) {
|
|
return NULL;
|
|
}
|
|
char *b = a;
|
|
while (*arg) {
|
|
char ch = *arg;
|
|
switch (ch) {
|
|
case '@':
|
|
case '`':
|
|
case '|':
|
|
case ';':
|
|
case '=':
|
|
case '\n':
|
|
break;
|
|
default:
|
|
*b++ = ch;
|
|
break;
|
|
}
|
|
arg++;
|
|
}
|
|
*b = 0;
|
|
return a;
|
|
}
|
|
|
|
static void pair_ut64x(PJ *pj, const char *key, ut64 val) {
|
|
if (pj) {
|
|
pair_ut64 (pj, key, val);
|
|
} else {
|
|
r_strf_var (sval, 32, "0x%"PFMT64x, val);
|
|
pair (key, sval);
|
|
}
|
|
}
|
|
|
|
static void pair_str(PJ *pj, const char *key, const char *val) {
|
|
if (pj) {
|
|
if (!val) {
|
|
val = "";
|
|
}
|
|
pj_ks (pj, key, val);
|
|
} else {
|
|
pair (key, val);
|
|
}
|
|
}
|
|
|
|
static ut64 rva(RBin *bin, ut64 paddr, ut64 vaddr, int va) {
|
|
if (va == VA_TRUE) {
|
|
if (paddr != UT64_MAX) {
|
|
return r_bin_get_vaddr (bin, paddr, vaddr);
|
|
}
|
|
}
|
|
if (va == VA_NOREBASE) {
|
|
return vaddr;
|
|
}
|
|
return paddr;
|
|
}
|
|
|
|
R_API bool r_core_bin_set_by_fd(RCore *core, ut64 bin_fd) {
|
|
if (r_bin_file_set_cur_by_fd (core->bin, bin_fd)) {
|
|
r_core_bin_set_cur (core, r_bin_cur (core->bin));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API void r_core_bin_export_info(RCore *core, int mode) {
|
|
char *flagname = NULL, *offset = NULL;
|
|
RBinFile *bf = r_bin_cur (core->bin);
|
|
if (!bf) {
|
|
return;
|
|
}
|
|
Sdb *db = sdb_ns (bf->sdb, "info", 0);;
|
|
if (!db) {
|
|
return;
|
|
}
|
|
SdbListIter *iter;
|
|
SdbKv *kv;
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("fs format\n");
|
|
} else if (IS_MODE_SET (mode)) {
|
|
r_flag_space_push (core->flags, "format");
|
|
}
|
|
// iterate over all keys
|
|
SdbList *ls = sdb_foreach_list (db, false);
|
|
ls_foreach (ls, iter, kv) {
|
|
char *k = sdbkv_key (kv);
|
|
char *v = sdbkv_value (kv);
|
|
char *dup = strdup (k);
|
|
//printf ("?e (%s) (%s)\n", k, v);
|
|
if ((flagname = strstr (dup, ".offset"))) {
|
|
*flagname = 0;
|
|
flagname = dup;
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("f %s @ %s\n", flagname, v);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
ut64 nv = r_num_math (core->num, v);
|
|
r_flag_set (core->flags, flagname, nv, 0);
|
|
}
|
|
free (offset);
|
|
offset = strdup (v);
|
|
}
|
|
if (strstr (dup, ".cparse")) {
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("\"td %s\"\n", v);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
char *code = r_str_newf ("%s;", v);
|
|
char *error_msg = NULL;
|
|
char *out = r_parse_c_string (core->anal, code, &error_msg);
|
|
free (code);
|
|
if (error_msg) {
|
|
eprintf ("%s", error_msg);
|
|
free (error_msg);
|
|
}
|
|
if (out) {
|
|
r_anal_save_parsed_type (core->anal, out);
|
|
free (out);
|
|
}
|
|
}
|
|
}
|
|
free (dup);
|
|
}
|
|
R_FREE (offset);
|
|
ls_foreach (ls, iter, kv) {
|
|
char *k = sdbkv_key (kv);
|
|
char *v = sdbkv_value (kv);
|
|
char *dup = strdup (k);
|
|
if ((flagname = strstr (dup, ".format"))) {
|
|
*flagname = 0;
|
|
if (!offset) {
|
|
offset = strdup ("0");
|
|
}
|
|
flagname = dup;
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("pf.%s %s\n", flagname, v);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
sdb_set (core->print->formats, flagname, v, 0);
|
|
}
|
|
}
|
|
free (dup);
|
|
}
|
|
ls_foreach (ls, iter, kv) {
|
|
char *k = sdbkv_key (kv);
|
|
char *v = sdbkv_value (kv);
|
|
char *dup = strdup (k);
|
|
if ((flagname = strstr (dup, ".format"))) {
|
|
*flagname = 0;
|
|
if (!offset) {
|
|
offset = strdup ("0");
|
|
}
|
|
flagname = dup;
|
|
int fmtsize = r_print_format_struct_size (core->print, v, 0, 0);
|
|
char *offset_key = r_str_newf ("%s.offset", flagname);
|
|
const char *off = sdb_const_get (db, offset_key, 0);
|
|
if (fmtsize < 1) {
|
|
continue;
|
|
}
|
|
free (offset_key);
|
|
if (off) {
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("Cf %d %s @ %s\n", fmtsize, v, off);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
ut64 addr = r_num_get (NULL, off);
|
|
ut8 *buf = malloc (fmtsize);
|
|
if (buf) {
|
|
r_io_read_at (core->io, addr, buf, fmtsize);
|
|
int res = r_print_format (core->print, addr, buf,
|
|
fmtsize, v, 0, NULL, NULL);
|
|
free (buf);
|
|
if (res < 0) {
|
|
eprintf ("Warning: Cannot register invalid format (%s)\n", v);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if ((flagname = strstr (dup, ".size"))) {
|
|
*flagname = 0;
|
|
flagname = dup;
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("fl %s %s\n", flagname, v);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
RFlagItem *fi = r_flag_get (core->flags, flagname);
|
|
if (fi) {
|
|
fi->size = r_num_math (core->num, v);
|
|
} else {
|
|
eprintf ("Cannot find flag named '%s'\n", flagname);
|
|
}
|
|
}
|
|
}
|
|
free (dup);
|
|
}
|
|
free (offset);
|
|
ls_free (ls);
|
|
if (IS_MODE_SET (mode)) {
|
|
r_flag_space_pop (core->flags);
|
|
}
|
|
}
|
|
|
|
R_API bool r_core_bin_load_structs(RCore *core, const char *file) {
|
|
r_return_val_if_fail (core && core->io, false);
|
|
if (!file) {
|
|
int fd = r_io_fd_get_current (core->io);
|
|
RIODesc *desc = r_io_desc_get (core->io, fd);
|
|
if (desc) {
|
|
file = desc->name;
|
|
}
|
|
if (!file) {
|
|
return false;
|
|
}
|
|
}
|
|
if (strchr (file, '\"')) { // TODO: escape "?
|
|
eprintf ("Invalid char found in filename\n");
|
|
return false;
|
|
}
|
|
RBinFileOptions opt = { 0 };
|
|
r_bin_open (core->bin, file, &opt);
|
|
RBinFile *bf = r_bin_cur (core->bin);
|
|
if (bf) {
|
|
r_core_bin_export_info (core, R_MODE_SET);
|
|
r_bin_file_delete (core->bin, bf->id);
|
|
return true;
|
|
}
|
|
eprintf ("Cannot open bin '%s'\n", file);
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_core_bin_set_by_name(RCore *core, const char *name) {
|
|
if (r_bin_file_set_cur_by_name (core->bin, name)) {
|
|
r_core_bin_set_cur (core, r_bin_cur (core->bin));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_core_bin_set_env(RCore *r, RBinFile *binfile) {
|
|
r_return_val_if_fail (r, false);
|
|
|
|
RBinObject *binobj = binfile? binfile->o: NULL;
|
|
RBinInfo *info = binobj? binobj->info: NULL;
|
|
if (info) {
|
|
int va = info->has_va;
|
|
const char *arch = info->arch;
|
|
ut16 bits = info->bits;
|
|
ut64 baseaddr = r_bin_get_baddr (r->bin);
|
|
r_config_set_i (r->config, "bin.baddr", baseaddr);
|
|
sdb_num_add (r->sdb, "orig_baddr", baseaddr, 0);
|
|
r->dbg->bp->baddr = baseaddr;
|
|
r_config_set (r->config, "asm.arch", arch);
|
|
r_config_set_i (r->config, "asm.bits", bits);
|
|
r_config_set (r->config, "anal.arch", arch);
|
|
if (info->cpu && *info->cpu) {
|
|
r_config_set (r->config, "anal.cpu", info->cpu);
|
|
} else {
|
|
r_config_set (r->config, "anal.cpu", arch);
|
|
}
|
|
r_asm_use (r->rasm, arch);
|
|
r_core_bin_info (r, R_CORE_BIN_ACC_ALL, NULL, R_MODE_SET, va, NULL, NULL);
|
|
r_core_bin_set_cur (r, binfile);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_core_bin_set_cur(RCore *core, RBinFile *binfile) {
|
|
r_return_val_if_fail (core && binfile, false);
|
|
ut32 fd = UT32_MAX;
|
|
if (!core->bin) {
|
|
return false;
|
|
}
|
|
if (!binfile) {
|
|
if (core && core->io->desc) {
|
|
fd = core->io->desc->fd;
|
|
}
|
|
}
|
|
// Find first available binfile
|
|
if (fd == UT32_MAX) {
|
|
return false;
|
|
}
|
|
binfile = r_bin_file_find_by_fd (core->bin, fd);
|
|
if (!binfile) {
|
|
return false;
|
|
}
|
|
r_bin_file_set_cur_binfile (core->bin, binfile);
|
|
return true;
|
|
}
|
|
|
|
static void _print_strings(RCore *r, RList *list, PJ *pj, int mode, int va) {
|
|
bool b64str = r_config_get_i (r->config, "bin.b64str");
|
|
int minstr = r_config_get_i (r->config, "bin.minstr");
|
|
int maxstr = r_config_get_i (r->config, "bin.maxstr");
|
|
RTable *table = r_core_table (r, "strings");
|
|
if (!table) {
|
|
return;
|
|
}
|
|
RBin *bin = r->bin;
|
|
RBinObject *obj = r_bin_cur_object (bin);
|
|
RListIter *iter;
|
|
RBinString *string;
|
|
RBinSection *section;
|
|
|
|
bin->minstrlen = minstr;
|
|
bin->maxstrlen = maxstr;
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_println ("fs strings");
|
|
} else if (IS_MODE_SET (mode) && r_config_get_i (r->config, "bin.strings")) {
|
|
r_flag_space_set (r->flags, R_FLAGS_FS_STRINGS);
|
|
r_cons_break_push (NULL, NULL);
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_println ("[Strings]");
|
|
r_table_set_columnsf (table, "nXXnnsss", "nth", "paddr", "vaddr", "len", "size", "section", "type", "string");
|
|
}
|
|
RBinString b64 = { 0 };
|
|
r_list_foreach (list, iter, string) {
|
|
const char *section_name, *type_string;
|
|
ut64 paddr, vaddr;
|
|
paddr = string->paddr;
|
|
vaddr = rva (r->bin, paddr, string->vaddr, va);
|
|
if (!r_bin_string_filter (bin, string->string, vaddr)) {
|
|
continue;
|
|
}
|
|
if (string->length < minstr) {
|
|
continue;
|
|
}
|
|
if (maxstr && string->length > maxstr) {
|
|
continue;
|
|
}
|
|
|
|
section = obj? r_bin_get_section_at (obj, paddr, 0): NULL;
|
|
section_name = section ? section->name : "";
|
|
type_string = r_bin_string_type (string->type);
|
|
if (b64str) {
|
|
ut8 *s = r_base64_decode_dyn (string->string, -1);
|
|
if (s && *s && IS_PRINTABLE (*s)) {
|
|
// TODO: add more checks
|
|
free (b64.string);
|
|
memcpy (&b64, string, sizeof (b64));
|
|
b64.string = (char *)s;
|
|
b64.size = strlen (b64.string);
|
|
string = &b64;
|
|
}
|
|
}
|
|
if (IS_MODE_SET (mode)) {
|
|
if (r_cons_is_breaked ()) {
|
|
break;
|
|
}
|
|
r_meta_set (r->anal, R_META_TYPE_STRING, vaddr, string->size, string->string);
|
|
char *str = (r->bin->prefix)
|
|
? r_str_newf ("%s.str.%s", r->bin->prefix, string->string)
|
|
: r_str_newf ("str.%s", string->string);
|
|
r_name_filter (str, -1);
|
|
RFlagItem *fi = r_flag_set (r->flags, str, vaddr, string->size);
|
|
const bool realstr = r_config_get_i (r->config, "bin.str.real");
|
|
if (fi && realstr) {
|
|
char *es = r_str_escape (string->string);
|
|
char *s = r_str_newf ("\"%s\"", es);
|
|
r_flag_item_set_realname (fi, s);
|
|
free (s);
|
|
free (es);
|
|
}
|
|
free (str);
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("0x%"PFMT64x" %d %d %s\n", vaddr,
|
|
string->size, string->length, string->string);
|
|
} else if (IS_MODE_SIMPLEST (mode)) {
|
|
r_cons_println (string->string);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
int *block_list;
|
|
pj_o (pj);
|
|
pj_kn (pj, "vaddr", vaddr);
|
|
pj_kn (pj, "paddr", paddr);
|
|
pj_kn (pj, "ordinal", string->ordinal);
|
|
pj_kn (pj, "size", string->size);
|
|
pj_kn (pj, "length", string->length);
|
|
pj_ks (pj, "section", section_name);
|
|
pj_ks (pj, "type", type_string);
|
|
// data itself may be encoded so use pj_ks
|
|
pj_ks (pj, "string", string->string);
|
|
|
|
switch (string->type) {
|
|
case R_STRING_TYPE_UTF8:
|
|
case R_STRING_TYPE_WIDE:
|
|
case R_STRING_TYPE_WIDE32:
|
|
block_list = r_utf_block_list ((const ut8*)string->string, -1, NULL);
|
|
if (block_list) {
|
|
if (block_list[0] == 0 && block_list[1] == -1) {
|
|
/* Don't include block list if
|
|
just Basic Latin (0x00 - 0x7F) */
|
|
R_FREE (block_list);
|
|
break;
|
|
}
|
|
int *block_ptr = block_list;
|
|
pj_k (pj, "blocks");
|
|
pj_a (pj);
|
|
for (; *block_ptr != -1; block_ptr++) {
|
|
const char *utfName = r_utf_block_name (*block_ptr);
|
|
pj_s (pj, r_str_get (utfName));
|
|
}
|
|
pj_end (pj);
|
|
R_FREE (block_list);
|
|
}
|
|
}
|
|
pj_end (pj);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
char *str = (r->bin->prefix)
|
|
? r_str_newf ("%s.str.%s", r->bin->prefix, string->string)
|
|
: r_str_newf ("str.%s", string->string);
|
|
r_name_filter (str, R_FLAG_NAME_SIZE);
|
|
r_cons_printf ("f %s %u 0x%08"PFMT64x"\n"
|
|
"Cs %u @ 0x%08"PFMT64x"\n",
|
|
str, string->size, vaddr,
|
|
string->size, vaddr);
|
|
free (str);
|
|
} else {
|
|
int *block_list;
|
|
char *str = string->string;
|
|
char *no_dbl_bslash_str = NULL;
|
|
if (!r->print->esc_bslash) {
|
|
char *ptr;
|
|
for (ptr = str; *ptr; ptr++) {
|
|
if (*ptr != '\\') {
|
|
continue;
|
|
}
|
|
if (*(ptr + 1) == '\\') {
|
|
if (!no_dbl_bslash_str) {
|
|
no_dbl_bslash_str = strdup (str);
|
|
if (!no_dbl_bslash_str) {
|
|
break;
|
|
}
|
|
ptr = no_dbl_bslash_str + (ptr - str);
|
|
}
|
|
memmove (ptr + 1, ptr + 2, strlen (ptr + 2) + 1);
|
|
}
|
|
}
|
|
if (no_dbl_bslash_str) {
|
|
str = no_dbl_bslash_str;
|
|
}
|
|
}
|
|
|
|
RStrBuf *buf = r_strbuf_new (str);
|
|
switch (string->type) {
|
|
case R_STRING_TYPE_UTF8:
|
|
case R_STRING_TYPE_WIDE:
|
|
case R_STRING_TYPE_WIDE32:
|
|
block_list = r_utf_block_list ((const ut8*)string->string, -1, NULL);
|
|
if (block_list) {
|
|
if (block_list[0] == 0 && block_list[1] == -1) {
|
|
/* Don't show block list if
|
|
just Basic Latin (0x00 - 0x7F) */
|
|
break;
|
|
}
|
|
int *block_ptr = block_list;
|
|
r_strbuf_append (buf, " blocks=");
|
|
for (; *block_ptr != -1; block_ptr++) {
|
|
if (block_ptr != block_list) {
|
|
r_strbuf_append (buf, ",");
|
|
}
|
|
const char *name = r_utf_block_name (*block_ptr);
|
|
r_strbuf_appendf (buf, "%s", r_str_get (name));
|
|
}
|
|
free (block_list);
|
|
}
|
|
break;
|
|
}
|
|
char *bufstr = r_strbuf_drain (buf);
|
|
r_table_add_rowf (table, "nXXddsss", (ut64)string->ordinal, paddr, vaddr,
|
|
(int)string->length, (int)string->size, section_name,
|
|
type_string, bufstr);
|
|
free (bufstr);
|
|
free (no_dbl_bslash_str);
|
|
}
|
|
}
|
|
R_FREE (b64.string);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
r_cons_break_pop ();
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
if (r->table_query) {
|
|
r_table_query (table, r->table_query);
|
|
}
|
|
char *s = r_table_tostring (table);
|
|
if (s) {
|
|
r_cons_print (s);
|
|
free (s);
|
|
}
|
|
}
|
|
r_table_free (table);
|
|
}
|
|
|
|
static bool bin_raw_strings(RCore *r, PJ *pj, int mode, int va) {
|
|
RBinFile *bf = r_bin_cur (r->bin);
|
|
bool new_bf = false;
|
|
if (bf && strstr (bf->file, "malloc://")) {
|
|
//sync bf->buf to search string on it
|
|
ut8 *tmp = R_NEWS (ut8, bf->size);
|
|
if (!tmp) {
|
|
return false;
|
|
}
|
|
r_io_read_at (r->io, 0, tmp, bf->size);
|
|
r_buf_write_at (bf->buf, 0, tmp, bf->size);
|
|
}
|
|
if (!r->io->desc) {
|
|
eprintf ("Core file not open\n");
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
pj_end (pj);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
if (!bf) {
|
|
bf = R_NEW0 (RBinFile);
|
|
if (!bf) {
|
|
return false;
|
|
}
|
|
RIODesc *desc = r->io->desc;
|
|
if (!desc) {
|
|
free (bf);
|
|
return false;
|
|
}
|
|
bf->file = strdup (desc->name);
|
|
bf->size = r_io_desc_size (desc);
|
|
if (bf->size == UT64_MAX) {
|
|
free (bf);
|
|
return false;
|
|
}
|
|
bf->buf = r_buf_new_with_io (&r->bin->iob, r->io->desc->fd);
|
|
bf->o = NULL;
|
|
bf->rbin = r->bin;
|
|
new_bf = true;
|
|
va = false;
|
|
}
|
|
RList *l = r_bin_raw_strings (bf, 0);
|
|
_print_strings (r, l, pj, mode, va);
|
|
r_list_free (l);
|
|
if (new_bf) {
|
|
r_buf_free (bf->buf);
|
|
bf->buf = NULL;
|
|
bf->id = -1;
|
|
r_bin_file_free (bf);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool bin_strings(RCore *r, PJ *pj, int mode, int va) {
|
|
RList *list;
|
|
RBinFile *binfile = r_bin_cur (r->bin);
|
|
RBinPlugin *plugin = r_bin_file_cur_plugin (binfile);
|
|
int rawstr = r_config_get_i (r->config, "bin.rawstr");
|
|
if (!binfile || !plugin) {
|
|
return false;
|
|
}
|
|
if (!r_config_get_i (r->config, "bin.strings")) {
|
|
return false;
|
|
}
|
|
if (plugin->info && plugin->name) {
|
|
if (strcmp (plugin->name, "any") == 0 && !rawstr) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
pj_end (pj);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
if (!(list = r_bin_get_strings (r->bin))) {
|
|
return false;
|
|
}
|
|
_print_strings (r, list, pj, mode, va);
|
|
return true;
|
|
}
|
|
|
|
static const char* get_compile_time(Sdb *binFileSdb) {
|
|
Sdb *info_ns = sdb_ns (binFileSdb, "info", false);
|
|
const char *timeDateStamp_string = sdb_const_get (info_ns,
|
|
"image_file_header.TimeDateStamp_string", 0);
|
|
return timeDateStamp_string;
|
|
}
|
|
|
|
static bool is_executable(RBinObject *obj) {
|
|
RListIter *it;
|
|
RBinSection* sec;
|
|
r_return_val_if_fail (obj, false);
|
|
if (obj->info && obj->info->arch) {
|
|
return true;
|
|
}
|
|
r_list_foreach (obj->sections, it, sec) {
|
|
if (sec->perm & R_PERM_X) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void sdb_concat_by_path(Sdb *s, const char *path) {
|
|
Sdb *db = sdb_new (0, path, 0);
|
|
sdb_merge (s, db);
|
|
sdb_close (db);
|
|
sdb_free (db);
|
|
}
|
|
|
|
static void load_types_from(RCore *core, const char *fmt, ...) {
|
|
const char *dir_prefix = r_config_get (core->config, "dir.prefix");
|
|
va_list ap;
|
|
va_start (ap, fmt);
|
|
char *s = r_str_newvf (fmt, ap);
|
|
SdbGperf *gp = r_anal_get_gperf_types (s);
|
|
if (gp) {
|
|
#if HAVE_GPERF
|
|
Sdb *gd = sdb_new0 ();
|
|
sdb_open_gperf (gd, gp);
|
|
sdb_reset (core->anal->sdb_types);
|
|
sdb_merge (core->anal->sdb_types, gd);
|
|
sdb_close (gd);
|
|
sdb_free (gd);
|
|
#endif
|
|
} else {
|
|
char *dbpath = r_str_newf ("%s/%s/%s.sdb", dir_prefix, R2_SDB_FCNSIGN, s);
|
|
if (r_file_exists (dbpath)) {
|
|
sdb_concat_by_path (core->anal->sdb_types, dbpath);
|
|
}
|
|
free (dbpath);
|
|
}
|
|
free (s);
|
|
va_end (ap);
|
|
}
|
|
|
|
R_API void r_core_anal_type_init(RCore *core) {
|
|
r_return_if_fail (core && core->anal);
|
|
int bits = core->rasm->bits;
|
|
Sdb *types = core->anal->sdb_types;
|
|
// make sure they are empty this is initializing
|
|
sdb_reset (types);
|
|
const char *anal_arch = r_config_get (core->config, "anal.arch");
|
|
const char *os = r_config_get (core->config, "asm.os");
|
|
// spaguetti ahead
|
|
|
|
load_types_from (core, "types");
|
|
load_types_from (core, "types-%s", anal_arch);
|
|
load_types_from (core, "types-%s", os);
|
|
load_types_from (core, "types-%d", bits);
|
|
load_types_from (core, "types-%s-%d", os, bits);
|
|
load_types_from (core, "types-%s-%d", anal_arch, bits);
|
|
load_types_from (core, "types-%s-%s", anal_arch, os);
|
|
load_types_from (core, "types-%s-%s-%d", anal_arch, os, bits);
|
|
}
|
|
|
|
R_API void r_core_anal_cc_init(RCore *core) {
|
|
const char *anal_arch = r_config_get (core->config, "anal.arch");
|
|
int bits = core->anal->bits;
|
|
if (!anal_arch) {
|
|
return;
|
|
}
|
|
#if HAVE_GPERF
|
|
char *k = r_str_newf ("cc_%s_%d", anal_arch, bits);
|
|
SdbGperf *gp = r_anal_get_gperf_cc (k);
|
|
free (k);
|
|
sdb_reset (core->anal->sdb_cc);
|
|
if (gp) {
|
|
Sdb *gd = sdb_new0 ();
|
|
sdb_open_gperf (gd, gp);
|
|
sdb_merge (core->anal->sdb_cc, gd);
|
|
sdb_close (gd);
|
|
sdb_free (gd);
|
|
}
|
|
{
|
|
// same as "tcc `arcc`"
|
|
char *s = r_reg_profile_to_cc (core->anal->reg);
|
|
if (s) {
|
|
if (!r_anal_cc_set (core->anal, s)) {
|
|
eprintf ("Warning: Invalid CC from reg profile.\n");
|
|
}
|
|
free (s);
|
|
} else {
|
|
eprintf ("Warning: Cannot derive CC from reg profile.\n");
|
|
}
|
|
}
|
|
#else
|
|
const char *dir_prefix = r_config_get (core->config, "dir.prefix");
|
|
char *dbpath = r_str_newf (R_JOIN_3_PATHS ("%s", R2_SDB_FCNSIGN, "cc-%s-%d.sdb"),
|
|
dir_prefix, anal_arch, bits);
|
|
char *dbhomepath = r_str_newf (R_JOIN_3_PATHS ("~", R2_HOME_SDB_FCNSIGN, "cc-%s-%d.sdb"),
|
|
anal_arch, bits);
|
|
Sdb *cc = core->anal->sdb_cc;
|
|
// Avoid sdb reloading
|
|
if (cc->path && (!strcmp (cc->path, dbpath) || !strcmp (cc->path, dbhomepath))) {
|
|
free (dbpath);
|
|
free (dbhomepath);
|
|
return;
|
|
}
|
|
sdb_reset (cc);
|
|
{
|
|
// same as "tcc `arcc`"
|
|
char *s = r_reg_profile_to_cc (core->anal->reg);
|
|
if (s) {
|
|
if (!r_anal_cc_set (core->anal, s)) {
|
|
eprintf ("Warning: Invalid CC from reg profile.\n");
|
|
}
|
|
free (s);
|
|
} else {
|
|
eprintf ("Warning: Cannot derive CC from reg profile.\n");
|
|
}
|
|
}
|
|
R_FREE (cc->path);
|
|
if (r_file_exists (dbhomepath)) {
|
|
sdb_concat_by_path (cc, dbhomepath);
|
|
cc->path = strdup (dbhomepath);
|
|
}
|
|
if (r_file_exists (dbpath)) {
|
|
sdb_concat_by_path (cc, dbpath);
|
|
cc->path = strdup (dbpath);
|
|
}
|
|
if (anal_arch && sdb_isempty (core->anal->sdb_cc)) {
|
|
eprintf ("Warning: Missing calling conventions for '%s'. Deriving it from the regprofile.\n", anal_arch);
|
|
}
|
|
free (dbpath);
|
|
free (dbhomepath);
|
|
#endif
|
|
}
|
|
|
|
static int bin_info(RCore *r, PJ *pj, int mode, ut64 laddr) {
|
|
int i, j, v;
|
|
char str[R_FLAG_NAME_SIZE];
|
|
RBinInfo *info = r_bin_get_info (r->bin);
|
|
RBinFile *bf = r_bin_cur (r->bin);
|
|
if (!bf) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
}
|
|
return false;
|
|
}
|
|
RBinObject *obj = bf->o;
|
|
|
|
if (!info || !obj) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
bool havecode = is_executable (obj) | (obj->entries != NULL);
|
|
const char *compiled = get_compile_time (bf->sdb);
|
|
|
|
if (IS_MODE_SET (mode)) {
|
|
r_config_set (r->config, "file.type", info->rclass);
|
|
r_config_set (r->config, "cfg.bigendian",
|
|
info->big_endian ? "true" : "false");
|
|
if (!info->rclass || strcmp (info->rclass, "fs")) {
|
|
if (info->lang) {
|
|
r_config_set (r->config, "bin.lang", info->lang);
|
|
}
|
|
r_config_set (r->config, "asm.os", info->os);
|
|
if (info->rclass && !strcmp (info->rclass, "pe")) {
|
|
r_config_set (r->config, "anal.cpp.abi", "msvc");
|
|
} else {
|
|
r_config_set (r->config, "anal.cpp.abi", "itanium");
|
|
}
|
|
r_config_set (r->config, "asm.arch", info->arch);
|
|
if (info->cpu && *info->cpu) {
|
|
r_config_set (r->config, "asm.cpu", info->cpu);
|
|
}
|
|
if (info->features && *info->features) {
|
|
r_config_set (r->config, "asm.features", info->features);
|
|
}
|
|
r_config_set (r->config, "anal.arch", info->arch);
|
|
snprintf (str, R_FLAG_NAME_SIZE, "%i", info->bits);
|
|
r_config_set (r->config, "asm.bits", str);
|
|
r_config_set (r->config, "asm.dwarf",
|
|
(R_BIN_DBG_STRIPPED & info->dbg_info) ? "false" : "true");
|
|
}
|
|
r_core_anal_type_init (r);
|
|
r_core_anal_cc_init (r);
|
|
if (info->default_cc && r_anal_cc_exist (r->anal, info->default_cc)) {
|
|
r_core_cmdf (r, "e anal.cc=%s", info->default_cc);
|
|
}
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("arch %s\n", info->arch);
|
|
if (info->cpu && *info->cpu) {
|
|
r_cons_printf ("cpu %s\n", info->cpu);
|
|
}
|
|
r_cons_printf ("bits %d\n", info->bits);
|
|
r_cons_printf ("os %s\n", info->os);
|
|
r_cons_printf ("endian %s\n", info->big_endian? "big": "little");
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
if (info->type && !strcmp (info->type, "fs")) {
|
|
r_cons_printf ("e file.type=fs\n");
|
|
r_cons_printf ("m /root %s 0\n", info->arch);
|
|
} else {
|
|
r_cons_printf ("e cfg.bigendian=%s\n"
|
|
"e asm.bits=%i\n"
|
|
"e asm.dwarf=%s\n",
|
|
r_str_bool (info->big_endian),
|
|
info->bits,
|
|
r_str_bool (R_BIN_DBG_STRIPPED &info->dbg_info));
|
|
int v = r_anal_archinfo (r->anal, R_ANAL_ARCHINFO_ALIGN);
|
|
r_cons_printf ("e asm.pcalign=%d\n", (v > 0)? v: 0);
|
|
if (info->lang && *info->lang) {
|
|
r_cons_printf ("e bin.lang=%s\n", info->lang);
|
|
}
|
|
if (info->rclass && *info->rclass) {
|
|
r_cons_printf ("e file.type=%s\n",
|
|
info->rclass);
|
|
}
|
|
if (info->os) {
|
|
r_cons_printf ("e asm.os=%s\n", info->os);
|
|
}
|
|
if (info->arch) {
|
|
r_cons_printf ("e asm.arch=%s\n", info->arch);
|
|
}
|
|
if (info->cpu && *info->cpu) {
|
|
r_cons_printf ("e asm.cpu=%s\n", info->cpu);
|
|
}
|
|
if (info->default_cc) {
|
|
r_cons_printf ("e anal.cc=%s", info->default_cc);
|
|
}
|
|
}
|
|
} else {
|
|
// XXX: if type is 'fs' show something different?
|
|
char *tmp_buf;
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
}
|
|
pair_str (pj, "arch", info->arch);
|
|
if (info->cpu && *info->cpu) {
|
|
pair_str (pj, "cpu", info->cpu);
|
|
}
|
|
pair_ut64x (pj, "baddr", r_bin_get_baddr (r->bin));
|
|
pair_ut64 (pj, "binsz", r_bin_get_size (r->bin));
|
|
pair_str (pj, "bintype", info->rclass);
|
|
pair_int (pj, "bits", info->bits);
|
|
pair_bool (pj, "canary", info->has_canary);
|
|
if (info->has_retguard != -1) {
|
|
pair_bool (pj, "retguard", info->has_retguard);
|
|
}
|
|
pair_str (pj, "class", info->bclass);
|
|
if (info->actual_checksum) {
|
|
/* computed checksum */
|
|
pair_str (pj, "cmp.csum", info->actual_checksum);
|
|
}
|
|
pair_str (pj, "compiled", compiled);
|
|
pair_str (pj, "compiler", info->compiler);
|
|
pair_bool (pj, "crypto", info->has_crypto);
|
|
pair_str (pj, "dbg_file", info->debug_file_name);
|
|
pair_str (pj, "endian", info->big_endian ? "big" : "little");
|
|
if (info->rclass && !strcmp (info->rclass, "mdmp")) {
|
|
tmp_buf = sdb_get (bf->sdb, "mdmp.flags", 0);
|
|
if (tmp_buf) {
|
|
pair_str (pj, "flags", tmp_buf);
|
|
free (tmp_buf);
|
|
}
|
|
}
|
|
pair_bool (pj, "havecode", havecode);
|
|
if (info->claimed_checksum) {
|
|
/* checksum specified in header */
|
|
pair_str (pj, "hdr.csum", info->claimed_checksum);
|
|
}
|
|
pair_str (pj, "guid", info->guid);
|
|
pair_str (pj, "intrp", info->intrp);
|
|
pair_ut64x (pj, "laddr", laddr);
|
|
pair_str (pj, "lang", info->lang);
|
|
pair_bool (pj, "linenum", R_BIN_DBG_LINENUMS & info->dbg_info);
|
|
pair_bool (pj, "lsyms", R_BIN_DBG_SYMS & info->dbg_info);
|
|
pair_str (pj, "machine", info->machine);
|
|
pair_bool (pj, "nx", info->has_nx);
|
|
pair_str (pj, "os", info->os);
|
|
if (info->rclass && !strcmp (info->rclass, "pe")) {
|
|
pair_bool (pj, "overlay", info->pe_overlay);
|
|
}
|
|
pair_str (pj, "cc", info->default_cc);
|
|
pair_bool (pj, "pic", info->has_pi);
|
|
pair_bool (pj, "relocs", R_BIN_DBG_RELOCS & info->dbg_info);
|
|
Sdb *sdb_info = sdb_ns (obj->kv, "info", false);
|
|
tmp_buf = sdb_get (sdb_info, "elf.relro", 0);
|
|
if (tmp_buf) {
|
|
pair_str (pj, "relro", tmp_buf);
|
|
free (tmp_buf);
|
|
}
|
|
pair_str (pj, "rpath", info->rpath);
|
|
if (info->rclass && !strcmp (info->rclass, "pe")) {
|
|
//this should be moved if added to mach0 (or others)
|
|
pair_bool (pj, "signed", info->signature);
|
|
}
|
|
pair_bool (pj, "sanitize", info->has_sanitizers);
|
|
pair_bool (pj, "static", r_bin_is_static (r->bin));
|
|
if (info->rclass && !strcmp (info->rclass, "mdmp")) {
|
|
v = sdb_num_get (bf->sdb, "mdmp.streams", 0);
|
|
if (v != -1) {
|
|
pair_int (pj, "streams", v);
|
|
}
|
|
}
|
|
pair_bool (pj, "stripped", R_BIN_DBG_STRIPPED & info->dbg_info);
|
|
pair_str (pj, "subsys", info->subsystem);
|
|
pair_bool (pj, "va", info->has_va);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ko (pj, "checksums");
|
|
}
|
|
for (i = 0; info->sum[i].type; i++) {
|
|
RBinHash *h = &info->sum[i];
|
|
ut64 hash = r_hash_name_to_bits (h->type);
|
|
RHash *rh = r_hash_new (true, hash);
|
|
ut8 *tmp = R_NEWS (ut8, h->to);
|
|
if (!tmp) {
|
|
return false;
|
|
}
|
|
r_buf_read_at (bf->buf, h->from, tmp, h->to);
|
|
int len = r_hash_calculate (rh, hash, tmp, h->to);
|
|
free (tmp);
|
|
if (len < 1) {
|
|
eprintf ("Invalid checksum length\n");
|
|
}
|
|
r_hash_free (rh);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ko (pj, h->type);
|
|
char *buf = malloc (2 * h->len + 1);
|
|
if (!buf) {
|
|
return false;
|
|
}
|
|
for (j = 0; j < h->len; j++) {
|
|
snprintf (buf + 2 * j, 3, "%02x", h->buf[j]);
|
|
}
|
|
pj_ks (pj, "hex", buf);
|
|
free (buf);
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_printf ("%s %" PFMT64u "-%" PFMT64u "c ", h->type, h->from, h->to+h->from);
|
|
for (j = 0; j < h->len; j++) {
|
|
r_cons_printf ("%02x", h->buf[j]);
|
|
}
|
|
r_cons_newline ();
|
|
}
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
const char *dir_prefix = r_config_get (r->config, "dir.prefix");
|
|
char spath[1024];
|
|
snprintf (spath, sizeof (spath), "%s/"R2_SDB_FCNSIGN"/spec.sdb", dir_prefix);
|
|
if (r_file_exists (spath)) {
|
|
sdb_concat_by_path (r->anal->sdb_fmts, spath);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
typedef struct {
|
|
size_t *line_starts;
|
|
char *content;
|
|
size_t line_count;
|
|
} FileLines;
|
|
|
|
static void file_lines_free(FileLines *file) {
|
|
if (!file) {
|
|
return;
|
|
}
|
|
free (file->line_starts);
|
|
free (file->content);
|
|
free (file);
|
|
}
|
|
|
|
FileLines *read_file_lines(const char *path) {
|
|
FileLines *result = R_NEW0 (FileLines);
|
|
if (!result) {
|
|
return result;
|
|
}
|
|
result->content = r_file_slurp (path, NULL);
|
|
if (result->content) {
|
|
result->line_starts = r_str_split_lines (result->content, &result->line_count);
|
|
}
|
|
if (!result->content || !result->line_starts) {
|
|
R_FREE (result);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static void file_lines_free_kv(HtPPKv *kv) {
|
|
free (kv->key);
|
|
file_lines_free (kv->value);
|
|
}
|
|
|
|
static bool bin_dwarf(RCore *core, PJ *pj, int mode) {
|
|
RBinDwarfRow *row;
|
|
RListIter *iter;
|
|
if (!r_config_get_i (core->config, "bin.dbginfo")) {
|
|
return false;
|
|
}
|
|
RBinFile *binfile = r_bin_cur (core->bin);
|
|
RBinPlugin * plugin = r_bin_file_cur_plugin (binfile);
|
|
if (!binfile) {
|
|
return false;
|
|
}
|
|
RList *list = NULL;
|
|
RList *ownlist = NULL;
|
|
if (plugin && plugin->lines) {
|
|
// list is not cloned to improve speed. avoid use after free
|
|
list = plugin->lines (binfile);
|
|
} else if (core->bin) {
|
|
// TODO: complete and speed-up support for dwarf
|
|
RBinDwarfDebugAbbrev *da = r_bin_dwarf_parse_abbrev (core->bin, mode);
|
|
if (!da) {
|
|
return false;
|
|
}
|
|
RBinDwarfDebugInfo *info = r_bin_dwarf_parse_info (da, core->bin, mode);
|
|
HtUP /*<offset, List *<LocListEntry>*/ *loc_table = r_bin_dwarf_parse_loc (core->bin, core->anal->bits / 8);
|
|
// I suppose there is no reason the parse it for a printing purposes
|
|
if (info && mode != R_MODE_PRINT) {
|
|
/* Should we do this by default? */
|
|
RAnalDwarfContext ctx = {
|
|
.info = info,
|
|
.loc = loc_table
|
|
};
|
|
r_anal_dwarf_process_info (core->anal, &ctx);
|
|
}
|
|
if (loc_table) {
|
|
if (mode == R_MODE_PRINT) {
|
|
r_bin_dwarf_print_loc (loc_table, core->anal->bits / 8, r_cons_printf);
|
|
}
|
|
r_bin_dwarf_free_loc (loc_table);
|
|
}
|
|
r_bin_dwarf_free_debug_info (info);
|
|
r_bin_dwarf_parse_aranges (core->bin, mode);
|
|
list = ownlist = r_bin_dwarf_parse_line (core->bin, mode);
|
|
r_bin_dwarf_free_debug_abbrev (da);
|
|
}
|
|
if (!list) {
|
|
return false;
|
|
}
|
|
|
|
r_cons_break_push (NULL, NULL);
|
|
/* cache file:line contents */
|
|
HtPP* file_lines = ht_pp_new (NULL, file_lines_free_kv, NULL);
|
|
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
}
|
|
|
|
SetP *set = set_p_new ();
|
|
//TODO we should need to store all this in sdb, or do a filecontentscache in libr/util
|
|
//XXX this whole thing has leaks
|
|
r_list_foreach (list, iter, row) {
|
|
if (r_cons_is_breaked ()) {
|
|
break;
|
|
}
|
|
if (mode) {
|
|
// TODO: use 'Cl' instead of CC
|
|
const char *path = row->file;
|
|
FileLines *current_lines = ht_pp_find (file_lines, path, NULL);
|
|
if (!current_lines) {
|
|
if (!set_p_contains (set, (void*)path)) {
|
|
set_p_add (set, (void*)path);
|
|
current_lines = read_file_lines (path);
|
|
if (!ht_pp_insert (file_lines, path, current_lines)) {
|
|
file_lines_free (current_lines);
|
|
current_lines = NULL;
|
|
}
|
|
}
|
|
}
|
|
char *line = NULL;
|
|
|
|
if (current_lines) {
|
|
int nl = row->line - 1;
|
|
if (nl >= 0 && nl < current_lines->line_count) {
|
|
line = strdup (current_lines->content + current_lines->line_starts[nl]);
|
|
}
|
|
}
|
|
if (line) {
|
|
r_str_filter (line, strlen (line));
|
|
line = r_str_replace (line, "\"", "\\\"", 1);
|
|
line = r_str_replace (line, "\\\\", "\\", 1);
|
|
}
|
|
bool chopPath = !r_config_get_i (core->config, "dir.dwarf.abspath");
|
|
char *file = strdup (row->file);
|
|
if (chopPath) {
|
|
const char *slash = r_str_lchr (file, '/');
|
|
if (slash) {
|
|
memmove (file, slash + 1, strlen (slash));
|
|
}
|
|
}
|
|
// TODO: implement internal : if ((mode & R_MODE_SET))
|
|
if ((mode & R_MODE_SET)) {
|
|
// TODO: use CL here.. but its not necessary.. so better not do anything imho
|
|
// r_core_cmdf (core, "CL %s:%d 0x%08"PFMT64x, file, (int)row->line, row->address);
|
|
#if 0
|
|
char *cmt = r_str_newf ("%s:%d %s", file, (int)row->line, r_str_get (line));
|
|
r_meta_set_string (core->anal, R_META_TYPE_COMMENT, row->address, cmt);
|
|
free (cmt);
|
|
#endif
|
|
} else if (IS_MODE_JSON(mode)) {
|
|
pj_a (pj);
|
|
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", "CC");
|
|
pj_ks (pj, "file", file);
|
|
pj_ki (pj, "line_num", (int) row->line);
|
|
pj_kn (pj, "addr", row->address);
|
|
pj_end (pj);
|
|
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", "CL");
|
|
pj_ks (pj, "file", file);
|
|
pj_ki (pj, "line_num", (int) row->line);
|
|
pj_ks (pj, "line", r_str_get (line));
|
|
pj_kn (pj, "addr", row->address);
|
|
pj_end (pj);
|
|
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_printf ("CL %s:%d 0x%08" PFMT64x "\n",
|
|
file, (int)row->line,
|
|
row->address);
|
|
r_cons_printf ("\"CC %s:%d %s\"@0x%" PFMT64x
|
|
"\n",
|
|
file, row->line,
|
|
r_str_get (line), row->address);
|
|
}
|
|
free (file);
|
|
free (line);
|
|
} else {
|
|
r_cons_printf ("0x%08" PFMT64x "\t%s\t%d\n",
|
|
row->address, row->file, row->line);
|
|
}
|
|
}
|
|
if (IS_MODE_JSON(mode)) {
|
|
pj_end (pj);
|
|
}
|
|
set_p_free (set);
|
|
r_cons_break_pop ();
|
|
ht_pp_free (file_lines);
|
|
r_list_free (ownlist);
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_core_pdb_info(RCore *core, const char *file, PJ *pj, int mode) {
|
|
r_return_val_if_fail (core && file, false);
|
|
|
|
ut64 baddr = r_config_get_i (core->config, "bin.baddr");
|
|
if (core->bin->cur && core->bin->cur->o && core->bin->cur->o->baddr) {
|
|
baddr = core->bin->cur->o->baddr;
|
|
} else if (baddr == UT64_MAX) {
|
|
eprintf ("Warning: Cannot find base address, flags will probably be misplaced\n");
|
|
baddr = 0LL;
|
|
}
|
|
|
|
RPdb pdb = R_EMPTY;
|
|
|
|
pdb.cb_printf = r_cons_printf;
|
|
if (!init_pdb_parser (&pdb, file)) {
|
|
return false;
|
|
}
|
|
if (!pdb.pdb_parse (&pdb)) {
|
|
eprintf ("pdb was not parsed\n");
|
|
pdb.finish_pdb_parse (&pdb);
|
|
return false;
|
|
}
|
|
|
|
switch (mode) {
|
|
case R_MODE_SET:
|
|
mode = 's';
|
|
r_core_cmd0 (core, ".iP*");
|
|
return true;
|
|
case R_MODE_JSON:
|
|
mode = 'j';
|
|
break;
|
|
case '*':
|
|
case 1:
|
|
mode = 'r';
|
|
break;
|
|
default:
|
|
mode = 'd'; // default
|
|
break;
|
|
}
|
|
if (mode == 'j') {
|
|
pj_o (pj);
|
|
}
|
|
|
|
pdb.print_types (&pdb, pj, mode);
|
|
pdb.print_gvars (&pdb, baddr, pj, mode);
|
|
// Save compound types into SDB
|
|
r_parse_pdb_types (core->anal, &pdb);
|
|
pdb.finish_pdb_parse (&pdb);
|
|
|
|
if (mode == 'j') {
|
|
pj_end (pj);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static ut64 srclineVal(const void *a) {
|
|
return r_str_hash64 (a);
|
|
}
|
|
|
|
static int bin_source(RCore *r, PJ *pj, int mode) {
|
|
RList *final_list = r_list_new ();
|
|
RBinFile * binfile = r->bin->cur;
|
|
|
|
if (!binfile) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
}
|
|
r_list_free (final_list);
|
|
return false;
|
|
}
|
|
|
|
SdbListIter *iter;
|
|
RListIter *iter2;
|
|
char* srcline;
|
|
SdbKv *kv;
|
|
SdbList *ls = sdb_foreach_list (binfile->sdb_addrinfo, false);
|
|
ls_foreach (ls, iter, kv) {
|
|
char *v = sdbkv_value (kv);
|
|
RList *list = r_str_split_list (v, "|", 0);
|
|
srcline = r_list_get_bottom (list);
|
|
if (srcline) {
|
|
if (!strstr (srcline, "0x")){
|
|
r_list_append (final_list, srcline);
|
|
}
|
|
}
|
|
r_list_free (list);
|
|
}
|
|
r_cons_printf ("[Source file]\n");
|
|
r_list_uniq_inplace (final_list, srclineVal);
|
|
r_list_foreach (final_list, iter2, srcline) {
|
|
r_cons_printf ("%s\n", srcline);
|
|
}
|
|
r_list_free (final_list);
|
|
return true;
|
|
}
|
|
|
|
static int bin_main(RCore *r, PJ *pj, int mode, int va) {
|
|
RBinAddr *binmain = r_bin_get_sym (r->bin, R_BIN_SYM_MAIN);
|
|
ut64 addr;
|
|
if (!binmain) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
}
|
|
return false;
|
|
}
|
|
addr = va ? r_bin_a2b (r->bin, binmain->vaddr) : binmain->paddr;
|
|
|
|
if (IS_MODE_SET (mode)) {
|
|
r_flag_space_set (r->flags, R_FLAGS_FS_SYMBOLS);
|
|
r_flag_set (r->flags, "main", addr, r->blocksize);
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("%"PFMT64d, addr);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("fs symbols\n");
|
|
r_cons_printf ("f main @ 0x%08"PFMT64x"\n", addr);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_kn (pj, "vaddr", addr);
|
|
pj_kn (pj, "paddr", binmain->paddr);
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_printf ("[Main]\n");
|
|
r_cons_printf ("vaddr=0x%08"PFMT64x" paddr=0x%08"PFMT64x"\n",
|
|
addr, binmain->paddr);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static inline bool is_initfini(RBinAddr *entry) {
|
|
switch (entry->type) {
|
|
case R_BIN_ENTRY_TYPE_INIT:
|
|
case R_BIN_ENTRY_TYPE_FINI:
|
|
case R_BIN_ENTRY_TYPE_PREINIT:
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
static int bin_entry(RCore *r, PJ *pj, int mode, ut64 laddr, int va, bool inifin) {
|
|
char str[R_FLAG_NAME_SIZE];
|
|
RList *entries = r_bin_get_entries (r->bin);
|
|
RListIter *iter;
|
|
RBinAddr *entry = NULL;
|
|
int i = 0, init_i = 0, fini_i = 0, preinit_i = 0;
|
|
ut64 baddr = r_bin_get_baddr (r->bin);
|
|
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("fs symbols\n");
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
if (inifin) {
|
|
r_cons_printf ("[Constructors]\n");
|
|
} else {
|
|
r_cons_printf ("[Entrypoints]\n");
|
|
}
|
|
}
|
|
|
|
r_list_foreach (entries, iter, entry) {
|
|
ut64 paddr = entry->paddr;
|
|
ut64 hpaddr = UT64_MAX;
|
|
ut64 hvaddr = UT64_MAX;
|
|
if (mode != R_MODE_SET) {
|
|
if (inifin) {
|
|
if (entry->type == R_BIN_ENTRY_TYPE_PROGRAM) {
|
|
continue;
|
|
}
|
|
} else {
|
|
if (entry->type != R_BIN_ENTRY_TYPE_PROGRAM) {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
if (entry->hpaddr) {
|
|
hpaddr = entry->hpaddr;
|
|
if (entry->hvaddr) {
|
|
hvaddr = rva (r->bin, hpaddr, entry->hvaddr, va);
|
|
}
|
|
}
|
|
ut64 at = rva (r->bin, paddr, entry->vaddr, va);
|
|
const char *type = r_bin_entry_type_string (entry->type);
|
|
if (!type) {
|
|
type = "unknown";
|
|
}
|
|
const char *hpaddr_key = (entry->type == R_BIN_ENTRY_TYPE_PROGRAM)
|
|
? "haddr"
|
|
: "hpaddr";
|
|
if (IS_MODE_SET (mode)) {
|
|
r_flag_space_set (r->flags, R_FLAGS_FS_SYMBOLS);
|
|
if (entry->type == R_BIN_ENTRY_TYPE_INIT) {
|
|
snprintf (str, R_FLAG_NAME_SIZE, "entry.init%i", init_i);
|
|
} else if (entry->type == R_BIN_ENTRY_TYPE_FINI) {
|
|
snprintf (str, R_FLAG_NAME_SIZE, "entry.fini%i", fini_i);
|
|
} else if (entry->type == R_BIN_ENTRY_TYPE_PREINIT) {
|
|
snprintf (str, R_FLAG_NAME_SIZE, "entry.preinit%i", preinit_i);
|
|
} else {
|
|
snprintf (str, R_FLAG_NAME_SIZE, "entry%i", i);
|
|
}
|
|
r_flag_set (r->flags, str, at, 1);
|
|
if (is_initfini (entry) && hvaddr != UT64_MAX) {
|
|
r_meta_set (r->anal, R_META_TYPE_DATA, hvaddr, entry->bits / 8, NULL);
|
|
}
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("0x%08"PFMT64x"\n", at);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_kn (pj, "vaddr", at);
|
|
pj_kn (pj, "paddr", paddr);
|
|
pj_kn (pj, "baddr", baddr);
|
|
pj_kn (pj, "laddr", laddr);
|
|
if (hvaddr != UT64_MAX) {
|
|
pj_kn (pj, "hvaddr", hvaddr);
|
|
}
|
|
pj_kn (pj, hpaddr_key, hpaddr);
|
|
pj_ks (pj, "type", type);
|
|
pj_end (pj);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
char *name = NULL;
|
|
if (entry->type == R_BIN_ENTRY_TYPE_INIT) {
|
|
name = r_str_newf ("entry.init%i", init_i);
|
|
} else if (entry->type == R_BIN_ENTRY_TYPE_FINI) {
|
|
name = r_str_newf ("entry.fini%i", fini_i);
|
|
} else if (entry->type == R_BIN_ENTRY_TYPE_PREINIT) {
|
|
name = r_str_newf ("entry.preinit%i", preinit_i);
|
|
} else {
|
|
name = r_str_newf ("entry%i", i);
|
|
}
|
|
char *n = __filterQuotedShell (name);
|
|
r_cons_printf ("\"f %s 1 0x%08"PFMT64x"\"\n", n, at);
|
|
r_cons_printf ("\"f %s_%s 1 0x%08"PFMT64x"\"\n", n, hpaddr_key, hpaddr);
|
|
r_cons_printf ("\"s %s\"\n", n);
|
|
free (n);
|
|
free (name);
|
|
} else {
|
|
r_cons_printf ("vaddr=0x%08"PFMT64x" paddr=0x%08"PFMT64x, at, paddr);
|
|
if (is_initfini (entry) && hvaddr != UT64_MAX) {
|
|
r_cons_printf (" hvaddr=0x%08"PFMT64x, hvaddr);
|
|
}
|
|
r_cons_printf (" %s=", hpaddr_key);
|
|
if (hpaddr == UT64_MAX) {
|
|
r_cons_printf ("%"PFMT64d, hpaddr);
|
|
} else {
|
|
r_cons_printf ("0x%08"PFMT64x, hpaddr);
|
|
}
|
|
if (entry->type == R_BIN_ENTRY_TYPE_PROGRAM && hvaddr != UT64_MAX) {
|
|
r_cons_printf (" hvaddr=0x%08"PFMT64x, hvaddr);
|
|
}
|
|
r_cons_printf (" type=%s\n", type);
|
|
}
|
|
if (entry->type == R_BIN_ENTRY_TYPE_INIT) {
|
|
init_i++;
|
|
} else if (entry->type == R_BIN_ENTRY_TYPE_FINI) {
|
|
fini_i++;
|
|
} else if (entry->type == R_BIN_ENTRY_TYPE_PREINIT) {
|
|
preinit_i++;
|
|
} else {
|
|
i++;
|
|
}
|
|
}
|
|
if (IS_MODE_SET (mode)) {
|
|
if (entry) {
|
|
ut64 at = rva (r->bin, entry->paddr, entry->vaddr, va);
|
|
r_core_seek (r, at, false);
|
|
}
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_printf ("\n%i entrypoints\n", init_i + fini_i + preinit_i + i);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static const char *bin_reloc_type_name(RBinReloc *reloc) {
|
|
#define CASE(T) case R_BIN_RELOC_ ## T: return reloc->additive ? "ADD_" #T : "SET_" #T
|
|
switch (reloc->type) {
|
|
CASE(1);
|
|
CASE(2);
|
|
CASE(4);
|
|
CASE(8);
|
|
CASE(16);
|
|
CASE(24);
|
|
CASE(32);
|
|
CASE(48);
|
|
CASE(64);
|
|
}
|
|
return "UNKNOWN";
|
|
#undef CASE
|
|
}
|
|
|
|
static ut8 bin_reloc_size(RBinReloc *reloc) {
|
|
#define CASE(T) case R_BIN_RELOC_ ## T: return (T) / 8
|
|
switch (reloc->type) {
|
|
CASE(1);
|
|
CASE(2);
|
|
CASE(4);
|
|
CASE(8);
|
|
CASE(16);
|
|
CASE(24);
|
|
CASE(32);
|
|
CASE(48);
|
|
CASE(64);
|
|
}
|
|
return 0;
|
|
#undef CASE
|
|
}
|
|
|
|
static char *resolveModuleOrdinal(Sdb *sdb, const char *module, int ordinal) {
|
|
r_strf_buffer(64);
|
|
Sdb *db = sdb;
|
|
char *foo = sdb_get (db, r_strf ("%d", ordinal), 0);
|
|
return (foo && *foo) ? foo : NULL;
|
|
}
|
|
|
|
// name can be optionally used to explicitly set the used base name (for example for demangling), otherwise the import name will be used.
|
|
static char *construct_reloc_name(R_NONNULL RBinReloc *reloc, R_NULLABLE const char *name) {
|
|
RStrBuf *buf = r_strbuf_new ("");
|
|
|
|
// (optional) libname_
|
|
if (reloc->import && reloc->import->libname) {
|
|
r_strbuf_appendf (buf, "%s_", reloc->import->libname);
|
|
} else if (reloc->symbol && reloc->symbol->libname) {
|
|
r_strbuf_appendf (buf, "%s_", reloc->symbol->libname);
|
|
}
|
|
|
|
// actual name
|
|
if (name) {
|
|
r_strbuf_append (buf, name);
|
|
} else if (reloc->import && reloc->import->name && *reloc->import->name) {
|
|
r_strbuf_append (buf, reloc->import->name);
|
|
} else if (reloc->symbol && reloc->symbol->name && *reloc->symbol->name) {
|
|
r_strbuf_appendf (buf, "%s", reloc->symbol->name);
|
|
} else if (reloc->is_ifunc) {
|
|
// addend is the function pointer for the resolving ifunc
|
|
r_strbuf_appendf (buf, "ifunc_%"PFMT64x, reloc->addend);
|
|
} else {
|
|
// TODO(eddyb) implement constant relocs.
|
|
r_strbuf_set (buf, "");
|
|
}
|
|
|
|
return r_strbuf_drain (buf);
|
|
}
|
|
|
|
static void set_bin_relocs(RCore *r, RBinReloc *reloc, ut64 addr, Sdb **db, char **sdb_module) {
|
|
bool bin_demangle = r_config_get_b (r->config, "bin.demangle");
|
|
bool keep_lib = r_config_get_b (r->config, "bin.demangle.libs");
|
|
const char *lang = r_config_get (r->config, "bin.lang");
|
|
bool is_pe = true;
|
|
int is_sandbox = r_sandbox_enable (0);
|
|
r_strf_buffer (64);
|
|
|
|
if (is_pe && !is_sandbox && reloc->import
|
|
&& reloc->import->name && reloc->import->libname
|
|
&& r_str_startswith (reloc->import->name, "Ordinal_")) {
|
|
char *module = strdup (reloc->import->libname);
|
|
r_str_case (module, false);
|
|
|
|
// strip trailing ".dll"
|
|
size_t module_len = strlen (module);
|
|
if (module_len > 4 && !strcmp (module + module_len - 4, ".dll")) {
|
|
module[module_len - 4] = '\0';
|
|
}
|
|
|
|
const char *import = reloc->import->name + strlen ("Ordinal_");
|
|
if (import) {
|
|
char *filename = NULL;
|
|
int ordinal = atoi (import);
|
|
if (!*sdb_module || strcmp (module, *sdb_module)) {
|
|
sdb_free (*db);
|
|
*db = NULL;
|
|
free (*sdb_module);
|
|
*sdb_module = strdup (module);
|
|
/* always lowercase */
|
|
filename = r_strf ("%s.sdb", module);
|
|
r_str_case (filename, false);
|
|
if (r_file_exists (filename)) {
|
|
*db = sdb_new (NULL, filename, 0);
|
|
} else {
|
|
const char *dirPrefix = r_sys_prefix (NULL);
|
|
filename = r_strf (R_JOIN_4_PATHS ("%s", R2_SDB_FORMAT, "dll", "%s.sdb"), dirPrefix, module);
|
|
if (r_file_exists (filename)) {
|
|
*db = sdb_new (NULL, filename, 0);
|
|
}
|
|
}
|
|
}
|
|
if (*db) {
|
|
// ordinal-1 because we enumerate starting at 0
|
|
char *symname = resolveModuleOrdinal (*db, module, ordinal - 1); // uses sdb_get
|
|
if (symname) {
|
|
if (r->bin->prefix) {
|
|
reloc->import->name = r_str_newf
|
|
("%s.%s", r->bin->prefix, symname);
|
|
R_FREE (symname);
|
|
} else {
|
|
reloc->import->name = symname;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
r_anal_hint_set_size (r->anal, reloc->vaddr, 4);
|
|
r_meta_set (r->anal, R_META_TYPE_DATA, reloc->vaddr, 4, NULL);
|
|
free (module);
|
|
}
|
|
|
|
char flagname[R_FLAG_NAME_SIZE];
|
|
char *reloc_name = construct_reloc_name (reloc, NULL);
|
|
if (!reloc_name || !*reloc_name) {
|
|
free (reloc_name);
|
|
return;
|
|
}
|
|
if (r->bin->prefix) {
|
|
snprintf (flagname, R_FLAG_NAME_SIZE, "%s.reloc.%s", r->bin->prefix, reloc_name);
|
|
} else {
|
|
snprintf (flagname, R_FLAG_NAME_SIZE, "reloc.%s", reloc_name);
|
|
}
|
|
free (reloc_name);
|
|
char *demname = NULL;
|
|
if (bin_demangle) {
|
|
demname = r_bin_demangle (r->bin->cur, lang, flagname, addr, keep_lib);
|
|
if (demname) {
|
|
snprintf (flagname, R_FLAG_NAME_SIZE, "reloc.%s", demname);
|
|
}
|
|
}
|
|
r_name_filter (flagname, 0);
|
|
RFlagItem *fi = r_flag_set (r->flags, flagname, addr, bin_reloc_size (reloc));
|
|
if (demname) {
|
|
char *realname = (r->bin->prefix)
|
|
? r_str_newf ("%s.reloc.%s", r->bin->prefix, demname)
|
|
: r_str_newf ("%s", demname);
|
|
r_flag_item_set_realname (fi, realname);
|
|
free (realname);
|
|
}
|
|
free (demname);
|
|
}
|
|
|
|
/* Define new data at relocation address if it's not in an executable section */
|
|
static void add_metadata(RCore *r, RBinReloc *reloc, ut64 addr, int mode) {
|
|
RBinFile * binfile = r->bin->cur;
|
|
RBinObject *binobj = binfile ? binfile->o: NULL;
|
|
RBinInfo *info = binobj ? binobj->info: NULL;
|
|
|
|
int cdsz = info? (info->bits == 64? 8: info->bits == 32? 4: info->bits == 16 ? 4: 0): 0;
|
|
if (cdsz == 0) {
|
|
return;
|
|
}
|
|
|
|
RIOMap *map = r_io_map_get_at (r->io, addr);
|
|
if (!map || map ->perm & R_PERM_X) {
|
|
return;
|
|
}
|
|
if (IS_MODE_SET (mode)) {
|
|
r_meta_set (r->anal, R_META_TYPE_DATA, reloc->vaddr, cdsz, NULL);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("Cd %d @ 0x%08" PFMT64x "\n", cdsz, addr);
|
|
}
|
|
}
|
|
|
|
static bool is_section_symbol(RBinSymbol *s) {
|
|
/* workaround for some bin plugs (e.g. ELF) */
|
|
if (!s || *s->name) {
|
|
return false;
|
|
}
|
|
return (s->type && !strcmp (s->type, R_BIN_TYPE_SECTION_STR));
|
|
}
|
|
|
|
static bool is_special_symbol(RBinSymbol *s) {
|
|
return s->type && !strcmp (s->type, R_BIN_TYPE_SPECIAL_SYM_STR);
|
|
}
|
|
|
|
static bool is_section_reloc(RBinReloc *r) {
|
|
return is_section_symbol (r->symbol);
|
|
}
|
|
|
|
static bool is_file_symbol(RBinSymbol *s) {
|
|
/* workaround for some bin plugs (e.g. ELF) */
|
|
return (s && s->type && !strcmp (s->type, R_BIN_TYPE_FILE_STR));
|
|
}
|
|
|
|
static bool is_file_reloc(RBinReloc *r) {
|
|
return is_file_symbol (r->symbol);
|
|
}
|
|
|
|
static int bin_relocs(RCore *r, PJ *pj, int mode, int va) {
|
|
bool bin_demangle = r_config_get_i (r->config, "bin.demangle");
|
|
bool keep_lib = r_config_get_i (r->config, "bin.demangle.libs");
|
|
const char *lang = r_config_get (r->config, "bin.lang");
|
|
RTable *table = r_core_table (r, "relocs");
|
|
r_return_val_if_fail (table, false);
|
|
Sdb *db = NULL;
|
|
char *sdb_module = NULL;
|
|
int i = 0;
|
|
|
|
R_TIME_PROFILE_BEGIN;
|
|
|
|
va = VA_TRUE; // XXX relocs always vaddr?
|
|
//this has been created for reloc object files
|
|
bool bin_cache = r_config_get_i (r->config, "bin.cache");
|
|
if (bin_cache) {
|
|
r_config_set (r->config, "io.cache", "true");
|
|
}
|
|
RRBTree *relocs = r_bin_patch_relocs (r->bin);
|
|
if (!relocs) {
|
|
if (bin_cache) {
|
|
r_config_set (r->config, "io.cache", "false");
|
|
bin_cache = false;
|
|
}
|
|
relocs = r_bin_get_relocs (r->bin);
|
|
}
|
|
if (bin_cache) {
|
|
if (r_pvector_len (&r->io->cache) == 0) {
|
|
r_config_set (r->config, "io.cache", "false");
|
|
} else {
|
|
r_config_set (r->config, "io.cache.read", "true");
|
|
}
|
|
}
|
|
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_println ("fs relocs");
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_println ("[Relocations]");
|
|
r_table_set_columnsf (table, "XXss", "vaddr", "paddr", "type", "name");
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
r_flag_space_set (r->flags, R_FLAGS_FS_RELOCS);
|
|
}
|
|
|
|
RRBNode *node;
|
|
RBinReloc *reloc;
|
|
r_crbtree_foreach (relocs, node, reloc) {
|
|
ut64 addr = rva (r->bin, reloc->paddr, reloc->vaddr, va);
|
|
if (IS_MODE_SET (mode) && (is_section_reloc (reloc) || is_file_reloc (reloc))) {
|
|
/*
|
|
* Skip section reloc because they will have their own flag.
|
|
* Skip also file reloc because not useful for now.
|
|
*/
|
|
} else if (IS_MODE_SET (mode)) {
|
|
set_bin_relocs (r, reloc, addr, &db, &sdb_module);
|
|
add_metadata (r, reloc, addr, mode);
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("0x%08"PFMT64x" %s\n", addr, reloc->import ? reloc->import->name : "");
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
char *name = reloc->import
|
|
? strdup (reloc->import->name)
|
|
: (reloc->symbol ? strdup (reloc->symbol->name) : NULL);
|
|
if (name && bin_demangle) {
|
|
char *mn = r_bin_demangle (r->bin->cur, NULL, name, addr, keep_lib);
|
|
if (mn) {
|
|
free (name);
|
|
name = mn;
|
|
}
|
|
}
|
|
if (name) {
|
|
int reloc_size = 4;
|
|
char *n = __filterQuotedShell (name);
|
|
r_cons_printf ("\"f %s%s%s %d 0x%08"PFMT64x"\"\n",
|
|
r_str_get_fail (r->bin->prefix, "reloc."),
|
|
r->bin->prefix ? "." : "", n, reloc_size, addr);
|
|
add_metadata (r, reloc, addr, mode);
|
|
free (n);
|
|
free (name);
|
|
}
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
char *mn = NULL;
|
|
char *relname = NULL;
|
|
|
|
if (reloc->import) {
|
|
mn = r_bin_demangle (r->bin->cur, lang, reloc->import->name, addr, keep_lib);
|
|
relname = strdup (reloc->import->name);
|
|
} else if (reloc->symbol) {
|
|
mn = r_bin_demangle (r->bin->cur, lang, reloc->symbol->name, addr, keep_lib);
|
|
relname = strdup (reloc->symbol->name);
|
|
}
|
|
|
|
// check if name is available
|
|
if (relname && *relname) {
|
|
pj_ks (pj, "name", relname);
|
|
}
|
|
pj_ks (pj, "demname", r_str_get (mn));
|
|
pj_ks (pj, "type", bin_reloc_type_name (reloc));
|
|
pj_kn (pj, "vaddr", reloc->vaddr);
|
|
pj_kn (pj, "paddr", reloc->paddr);
|
|
if (reloc->symbol) {
|
|
pj_kn (pj, "sym_va", reloc->symbol->vaddr);
|
|
}
|
|
pj_kb (pj, "is_ifunc", reloc->is_ifunc);
|
|
// end reloc item
|
|
pj_end (pj);
|
|
|
|
free (mn);
|
|
if (relname) {
|
|
free (relname);
|
|
}
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
char *name = reloc->import
|
|
? strdup (reloc->import->name)
|
|
: reloc->symbol
|
|
? strdup (reloc->symbol->name)
|
|
: NULL;
|
|
if (bin_demangle) {
|
|
char *mn = r_bin_demangle (r->bin->cur, NULL, name, addr, keep_lib);
|
|
if (mn && *mn) {
|
|
free (name);
|
|
name = mn;
|
|
}
|
|
}
|
|
char *reloc_name = construct_reloc_name (reloc, name);
|
|
RStrBuf *buf = r_strbuf_new (reloc_name);
|
|
free (reloc_name);
|
|
R_FREE (name);
|
|
if (reloc->addend) {
|
|
if ((reloc->import || reloc->symbol) && !r_strbuf_is_empty (buf) && reloc->addend > 0) {
|
|
r_strbuf_append (buf," +");
|
|
}
|
|
if (reloc->addend < 0) {
|
|
r_strbuf_appendf (buf," - 0x%08"PFMT64x, -reloc->addend);
|
|
} else {
|
|
r_strbuf_appendf (buf," 0x%08"PFMT64x, reloc->addend);
|
|
}
|
|
}
|
|
if (reloc->is_ifunc) {
|
|
r_strbuf_append (buf, " (ifunc)");
|
|
}
|
|
char *res = r_strbuf_drain (buf);
|
|
r_table_add_rowf (table, "XXss", addr, reloc->paddr,
|
|
bin_reloc_type_name (reloc), res);
|
|
free (res);
|
|
}
|
|
i++;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
}
|
|
if (IS_MODE_NORMAL (mode)) {
|
|
if (r->table_query) {
|
|
r_table_query (table, r->table_query);
|
|
}
|
|
char *s = r_table_tostring (table);
|
|
r_cons_printf ("\n%s\n", s);
|
|
free (s);
|
|
r_cons_printf ("\n%i relocations\n", i);
|
|
|
|
}
|
|
|
|
r_table_free (table);
|
|
R_FREE (sdb_module);
|
|
sdb_free (db);
|
|
db = NULL;
|
|
|
|
R_TIME_PROFILE_END;
|
|
if (IS_MODE_JSON (mode) && relocs == NULL) {
|
|
return true;
|
|
}
|
|
return relocs != NULL;
|
|
}
|
|
|
|
#define MYDB 1
|
|
/* this is a VERY VERY VERY hacky and bad workaround that needs proper refactoring in Rbin to use Sdb */
|
|
#if MYDB
|
|
R_DEPRECATE static Sdb *mydb = NULL;
|
|
R_DEPRECATE static RList *osymbols = NULL;
|
|
|
|
R_DEPRECATE static RBinSymbol *get_import(RBin *bin, RList *symbols, const char *name, ut64 addr) {
|
|
r_strf_buffer(64);
|
|
RBinSymbol *symbol, *res = NULL;
|
|
RListIter *iter;
|
|
if (mydb && symbols && symbols != osymbols) {
|
|
sdb_free (mydb);
|
|
mydb = NULL;
|
|
osymbols = symbols;
|
|
}
|
|
if (mydb) {
|
|
if (name) {
|
|
res = (RBinSymbol*)(void*)(size_t)
|
|
sdb_num_get (mydb, r_strf ("%x", sdb_hash (name)), NULL);
|
|
} else {
|
|
res = (RBinSymbol*)(void*)(size_t)
|
|
sdb_num_get (mydb, r_strf ("0x%08"PFMT64x, addr), NULL);
|
|
}
|
|
} else {
|
|
mydb = sdb_new0 ();
|
|
r_list_foreach (symbols, iter, symbol) {
|
|
if (!symbol->name || !symbol->is_imported) {
|
|
continue;
|
|
}
|
|
/* ${name}=${ptrToSymbol} */
|
|
if (!sdb_num_add (mydb, r_strf ("%x", sdb_hash (symbol->name)), (ut64)(size_t)symbol, 0)) {
|
|
// eprintf ("DUP (%s)\n", symbol->name);
|
|
}
|
|
/* 0x${vaddr}=${ptrToSymbol} */
|
|
if (!sdb_num_add (mydb, r_strf ("0x%08"PFMT64x, symbol->vaddr), (ut64)(size_t)symbol, 0)) {
|
|
// eprintf ("DUP (%s)\n", symbol->name);
|
|
}
|
|
if (name) {
|
|
if (!res && !strcmp (symbol->name, name)) {
|
|
res = symbol;
|
|
}
|
|
} else {
|
|
if (symbol->vaddr == addr) {
|
|
res = symbol;
|
|
}
|
|
}
|
|
}
|
|
osymbols = symbols;
|
|
}
|
|
return res;
|
|
}
|
|
#else
|
|
static RList *osymbols = NULL;
|
|
static RBinSymbol *get_symbol(RBin *bin, RList *symbols, const char *name, ut64 addr) {
|
|
RBinSymbol *symbol;
|
|
RListIter *iter;
|
|
// XXX this is slow, we should use a hashtable here
|
|
r_list_foreach (symbols, iter, symbol) {
|
|
if (name) {
|
|
if (!strcmp (symbol->name, name))
|
|
return symbol;
|
|
} else {
|
|
if (symbol->vaddr == addr) {
|
|
return symbol;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
/* XXX: This is a hack to get PLT references in rabin2 -i */
|
|
R_API ut64 r_core_bin_impaddr(RBin *bin, int va, const char *name) {
|
|
RList *symbols;
|
|
|
|
if (!name || !*name) {
|
|
return false;
|
|
}
|
|
if (!(symbols = r_bin_get_symbols (bin))) {
|
|
return false;
|
|
}
|
|
RBinSymbol *s = get_import (bin, symbols, name, 0LL);
|
|
// maybe ut64_MAX to indicate import not found?
|
|
ut64 addr = 0LL;
|
|
if (s) {
|
|
if (va) {
|
|
if (s->paddr == UT64_MAX) {
|
|
addr = s->vaddr;
|
|
} else {
|
|
addr = r_bin_get_vaddr (bin, s->paddr, s->vaddr);
|
|
}
|
|
} else {
|
|
addr = s->paddr;
|
|
}
|
|
}
|
|
return addr;
|
|
}
|
|
|
|
static int bin_imports(RCore *r, PJ *pj, int mode, int va, const char *name) {
|
|
RBinInfo *info = r_bin_get_info (r->bin);
|
|
int bin_demangle = r_config_get_i (r->config, "bin.demangle");
|
|
bool keep_lib = r_config_get_b (r->config, "bin.demangle.libs");
|
|
RTable *table = r_core_table (r, "imports");
|
|
r_return_val_if_fail (table, false);
|
|
RBinImport *import;
|
|
RListIter *iter;
|
|
r_strf_buffer (64);
|
|
bool lit = info ? info->has_lit: false;
|
|
int i = 0;
|
|
|
|
if (!info) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
pj_end (pj);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
const RList *imports = r_bin_get_imports (r->bin);
|
|
int cdsz = info? (info->bits == 64? 8: info->bits == 32? 4: info->bits == 16 ? 4: 0): 0;
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_println ("fs imports");
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_println ("[Imports]");
|
|
r_table_set_columnsf (table, "nXssss", "nth", "vaddr", "bind", "type", "lib", "name");
|
|
}
|
|
r_list_foreach (imports, iter, import) {
|
|
if (name && strcmp (import->name, name)) {
|
|
continue;
|
|
}
|
|
char *symname = strdup (import->name);
|
|
char *libname = import->libname ? strdup (import->libname) : NULL;
|
|
ut64 addr = lit ? r_core_bin_impaddr (r->bin, va, symname): 0;
|
|
if (bin_demangle) {
|
|
char *dname = r_bin_demangle (r->bin->cur, NULL, symname, addr, keep_lib);
|
|
if (dname) {
|
|
free (symname);
|
|
symname = r_str_newf ("sym.imp.%s", dname);
|
|
free (dname);
|
|
}
|
|
}
|
|
if (r->bin->prefix) {
|
|
char *prname = r_str_newf ("%s.%s", r->bin->prefix, symname);
|
|
free (symname);
|
|
symname = prname;
|
|
}
|
|
if (IS_MODE_SET (mode)) {
|
|
// TODO(eddyb) symbols that are imports.
|
|
// Add a dword/qword for PE imports
|
|
if (libname && strstr (libname, ".dll") && cdsz) {
|
|
r_meta_set (r->anal, R_META_TYPE_DATA, addr, cdsz, NULL);
|
|
}
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("%s%s%s\n",
|
|
r_str_get (libname), libname ? " " : "", symname);
|
|
} else if (IS_MODE_SIMPLEST (mode)) {
|
|
r_cons_println (symname);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
|
|
pj_ki (pj, "ordinal", import->ordinal);
|
|
if (import->bind) {
|
|
pj_ks (pj, "bind", import->bind);
|
|
}
|
|
if (import->type) {
|
|
pj_ks (pj, "type", import->type);
|
|
}
|
|
if (import->classname && import->classname[0]) {
|
|
pj_ks (pj, "classname", import->classname);
|
|
pj_ks (pj, "descriptor", import->descriptor);
|
|
}
|
|
pj_ks (pj, "name", symname);
|
|
if (libname) {
|
|
pj_ks (pj, "libname", libname);
|
|
}
|
|
pj_kn (pj, "plt", addr);
|
|
pj_end (pj);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
// TODO(eddyb) symbols that are imports.
|
|
} else {
|
|
const char *bind = r_str_get_fail (import->bind, "NONE");
|
|
const char *type = r_str_get_fail (import->type, "NONE");
|
|
if (import->classname && import->classname[0]) {
|
|
r_table_add_rowf (table, "nXssss", (ut64)import->ordinal, addr, bind, type, r_str_get (libname),
|
|
r_strf ("%s.%s", import->classname, symname));
|
|
} else {
|
|
r_table_add_rowf (table, "nXssss", (ut64)import->ordinal, addr, bind, type, r_str_get (libname),
|
|
symname);
|
|
}
|
|
|
|
if (import->descriptor && import->descriptor[0]) {
|
|
// Uh?
|
|
r_cons_printf (" descriptor=%s", import->descriptor);
|
|
}
|
|
if (!IS_MODE_NORMAL (mode)) {
|
|
r_cons_newline ();
|
|
}
|
|
}
|
|
R_FREE (symname);
|
|
R_FREE (libname);
|
|
i++;
|
|
}
|
|
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
if (r->table_query) {
|
|
r_table_query (table, r->table_query);
|
|
}
|
|
char *s = r_table_tostring (table);
|
|
r_cons_printf ("%s\n", s);
|
|
free (s);
|
|
}
|
|
|
|
r_table_free (table);
|
|
#if MYDB
|
|
// NOTE: if we comment out this, it will leak.. but it will be faster
|
|
// because it will keep the cache across multiple RBin calls
|
|
osymbols = NULL;
|
|
sdb_free (mydb);
|
|
mydb = NULL;
|
|
#endif
|
|
return true;
|
|
}
|
|
|
|
static const char *getPrefixFor(RBinSymbol *sym) {
|
|
if (sym) {
|
|
// workaround for ELF
|
|
if (sym->type) {
|
|
if (!strcmp (sym->type, R_BIN_TYPE_NOTYPE_STR)) {
|
|
return sym->is_imported ? "loc.imp" : "loc";
|
|
}
|
|
if (!strcmp (sym->type, R_BIN_TYPE_OBJECT_STR)) {
|
|
return sym->is_imported ? "obj.imp" : "obj";
|
|
}
|
|
}
|
|
return sym->is_imported ? "sym.imp" : "sym";
|
|
}
|
|
return "sym";
|
|
}
|
|
|
|
#define MAXFLAG_LEN_DEFAULT 128
|
|
|
|
static char *construct_symbol_flagname(const char *pfx, const char *libname, const char *symname, int len) {
|
|
char *r = r_str_newf ("%s.%s%s%s", pfx, r_str_get (libname), libname ? "_" : "", symname);
|
|
if (r) {
|
|
r_name_filter (r, len); // maybe unnecessary..
|
|
char *R = __filterQuotedShell (r);
|
|
free (r);
|
|
return R;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
typedef struct {
|
|
const char *pfx; // prefix for flags
|
|
char *name; // raw symbol name
|
|
char *libname; // name of the lib this symbol is specific to, if any
|
|
char *nameflag; // flag name for symbol
|
|
char *demname; // demangled raw symbol name
|
|
char *demflag; // flag name for demangled symbol
|
|
char *classname; // classname
|
|
char *classflag; // flag for classname
|
|
char *methname; // methods [class]::[method]
|
|
char *methflag; // methods flag sym.[class].[method]
|
|
} SymName;
|
|
|
|
static void snInit(RCore *r, SymName *sn, RBinSymbol *sym, const char *lang) {
|
|
int bin_demangle = lang != NULL;
|
|
bool keep_lib = r_config_get_b (r->config, "bin.demangle.libs");
|
|
if (!r || !sym || !sym->name) {
|
|
return;
|
|
}
|
|
sn->name = r_str_newf ("%s%s", sym->is_imported ? "imp." : "", sym->name);
|
|
sn->libname = sym->libname ? strdup (sym->libname) : NULL;
|
|
const char *pfx = getPrefixFor (sym);
|
|
const char *symname = sym->name;
|
|
char *resymname = NULL;
|
|
if (sym->dup_count > 0) {
|
|
resymname = r_str_newf ("%s_%d", sym->name, sym->dup_count);
|
|
symname = resymname;
|
|
}
|
|
sn->nameflag = construct_symbol_flagname (pfx, sym->libname, symname, MAXFLAG_LEN_DEFAULT);
|
|
free (resymname);
|
|
if (sym->classname && sym->classname[0]) {
|
|
sn->classname = strdup (sym->classname);
|
|
sn->classflag = r_str_newf ("sym.%s.%s", sn->classname, sn->name);
|
|
r_name_filter (sn->classflag, MAXFLAG_LEN_DEFAULT);
|
|
const char *name = sym->dname? sym->dname: sym->name;
|
|
sn->methname = r_str_newf ("%s::%s", sn->classname, name);
|
|
sn->methflag = r_str_newf ("sym.%s.%s", sn->classname, name);
|
|
r_name_filter (sn->methflag, strlen (sn->methflag));
|
|
} else {
|
|
sn->classname = NULL;
|
|
sn->classflag = NULL;
|
|
sn->methname = NULL;
|
|
sn->methflag = NULL;
|
|
}
|
|
sn->demname = NULL;
|
|
sn->demflag = NULL;
|
|
if (bin_demangle && sym->paddr) {
|
|
sn->demname = r_bin_demangle (r->bin->cur, lang, sn->name, sym->vaddr, keep_lib);
|
|
if (sn->demname) {
|
|
sn->demflag = construct_symbol_flagname (pfx, sym->libname, sn->demname, -1);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void snFini(SymName *sn) {
|
|
R_FREE (sn->name);
|
|
R_FREE (sn->libname);
|
|
R_FREE (sn->nameflag);
|
|
R_FREE (sn->demname);
|
|
R_FREE (sn->demflag);
|
|
R_FREE (sn->classname);
|
|
R_FREE (sn->classflag);
|
|
R_FREE (sn->methname);
|
|
R_FREE (sn->methflag);
|
|
}
|
|
|
|
static bool isAnExport(RBinSymbol *s) {
|
|
/* workaround for some bin plugs */
|
|
if (s->is_imported) {
|
|
return false;
|
|
}
|
|
return (s->bind && !strcmp (s->bind, R_BIN_BIND_GLOBAL_STR));
|
|
}
|
|
|
|
static ut64 compute_addr(RBin *bin, ut64 paddr, ut64 vaddr, int va) {
|
|
return paddr == UT64_MAX? vaddr: rva (bin, paddr, vaddr, va);
|
|
}
|
|
|
|
static void handle_arm_special_symbol(RCore *core, RBinSymbol *symbol, int va) {
|
|
ut64 addr = compute_addr (core->bin, symbol->paddr, symbol->vaddr, va);
|
|
if (!strcmp (symbol->name, "$a")) {
|
|
r_anal_hint_set_bits (core->anal, addr, 32);
|
|
} else if (!strcmp (symbol->name, "$x")) {
|
|
r_anal_hint_set_bits (core->anal, addr, 64);
|
|
} else if (!strcmp (symbol->name, "$t")) {
|
|
r_anal_hint_set_bits (core->anal, addr, 16);
|
|
} else if (!strcmp (symbol->name, "$d")) {
|
|
// TODO: we could add data meta type at addr, but sometimes $d
|
|
// is in the middle of the code and it would make the code less
|
|
// readable.
|
|
} else {
|
|
if (core->bin->verbose) {
|
|
R_LOG_WARN ("Special symbol %s not handled\n", symbol->name);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void handle_arm_hint(RCore *core, RBinInfo *info, ut64 paddr, ut64 vaddr, int bits, int va) {
|
|
if (info->bits > 32) { // we look at 16 or 32 bit only
|
|
return;
|
|
}
|
|
|
|
int force_bits = 0;
|
|
ut64 addr = compute_addr (core->bin, paddr, vaddr, va);
|
|
if (paddr & 1 || bits == 16) {
|
|
force_bits = 16;
|
|
} else if (info->bits == 16 && bits == 32) {
|
|
force_bits = 32;
|
|
} else if (!(paddr & 1) && bits == 32) {
|
|
force_bits = 32;
|
|
}
|
|
if (force_bits) {
|
|
r_anal_hint_set_bits (core->anal, addr, force_bits);
|
|
}
|
|
}
|
|
|
|
static void handle_arm_symbol(RCore *core, RBinSymbol *symbol, RBinInfo *info, int va) {
|
|
handle_arm_hint (core, info, symbol->paddr, symbol->vaddr, symbol->bits, va);
|
|
}
|
|
|
|
static void handle_arm_entry(RCore *core, RBinAddr *entry, RBinInfo *info, int va) {
|
|
handle_arm_hint (core, info, entry->paddr, entry->vaddr, entry->bits, va);
|
|
}
|
|
|
|
static void select_flag_space(RCore *core, RBinSymbol *symbol) {
|
|
if (symbol->is_imported) {
|
|
r_flag_space_push (core->flags, R_FLAGS_FS_IMPORTS);
|
|
} else if (symbol->type && !strcmp (symbol->type, R_BIN_TYPE_SECTION_STR)) {
|
|
r_flag_space_push (core->flags, R_FLAGS_FS_SYMBOLS_SECTIONS);
|
|
} else {
|
|
r_flag_space_push (core->flags, R_FLAGS_FS_SYMBOLS);
|
|
}
|
|
}
|
|
|
|
static int bin_symbols(RCore *r, PJ *pj, int mode, ut64 laddr, int va, ut64 at, const char *name, bool exponly, const char *args) {
|
|
RBinInfo *info = r_bin_get_info (r->bin);
|
|
RList *entries = r_bin_get_entries (r->bin);
|
|
RBinSymbol *symbol;
|
|
RBinAddr *entry;
|
|
RListIter *iter;
|
|
bool firstexp = true;
|
|
bool printHere = (args && *args == '.');
|
|
bool none = true;
|
|
|
|
int i = 0, lastfs = 's';
|
|
RTable *table = r_core_table (r, "symbols");
|
|
bool bin_demangle = r_config_get_i (r->config, "bin.demangle");
|
|
if (IS_MODE_JSON (mode)) {
|
|
if (!printHere) {
|
|
pj_a (pj);
|
|
}
|
|
}
|
|
if (!info) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
if (printHere) {
|
|
pj_o (pj);
|
|
}
|
|
pj_end (pj);
|
|
}
|
|
r_table_free (table);
|
|
return 0;
|
|
}
|
|
|
|
bool is_arm = info && info->arch && !strncmp (info->arch, "arm", 3);
|
|
const char *lang = bin_demangle ? r_config_get (r->config, "bin.lang") : NULL;
|
|
|
|
RList *symbols = r_bin_get_symbols (r->bin);
|
|
r_spaces_push (&r->anal->meta_spaces, "bin");
|
|
|
|
if (IS_MODE_SET (mode)) {
|
|
r_flag_space_set (r->flags, R_FLAGS_FS_SYMBOLS);
|
|
} else if (at == UT64_MAX && exponly) {
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("fs exports\n");
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_printf (printHere ? "" : "[Exports]\n");
|
|
}
|
|
} else if (at == UT64_MAX && !exponly) {
|
|
if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("fs symbols\n");
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_printf (printHere ? "" : "[Symbols]\n");
|
|
}
|
|
}
|
|
if (IS_MODE_NORMAL (mode)) {
|
|
r_table_set_columnsf (table, "dXXssdss", "nth", "paddr","vaddr","bind", "type", "size", "lib", "name");
|
|
}
|
|
|
|
size_t count = 0;
|
|
r_list_foreach (symbols, iter, symbol) {
|
|
if (!symbol->name) {
|
|
continue;
|
|
}
|
|
if (exponly && !isAnExport (symbol)) {
|
|
continue;
|
|
}
|
|
if (name && strcmp (symbol->name, name)) {
|
|
continue;
|
|
}
|
|
ut64 addr = compute_addr (r->bin, symbol->paddr, symbol->vaddr, va);
|
|
ut32 len = symbol->size ? symbol->size : 32;
|
|
if (at != UT64_MAX && (!symbol->size || !is_in_range (at, addr, symbol->size))) {
|
|
continue;
|
|
}
|
|
if ((printHere && !is_in_range (r->offset, symbol->paddr, len))
|
|
&& (printHere && !is_in_range (r->offset, addr, len))) {
|
|
continue;
|
|
}
|
|
SymName sn = {0};
|
|
count ++;
|
|
snInit (r, &sn, symbol, lang);
|
|
char *r_symbol_name = r_str_escape_utf8 (sn.name, false, true);
|
|
|
|
if (IS_MODE_SET (mode) && (is_section_symbol (symbol) || is_file_symbol (symbol))) {
|
|
/*
|
|
* Skip section symbols because they will have their own flag.
|
|
* Skip also file symbols because not useful for now.
|
|
*/
|
|
} else if (IS_MODE_SET (mode) && is_special_symbol (symbol)) {
|
|
if (is_arm) {
|
|
handle_arm_special_symbol (r, symbol, va);
|
|
}
|
|
} else if (IS_MODE_SET (mode)) {
|
|
// TODO: provide separate API in RBinPlugin to let plugins handle anal hints/metadata
|
|
if (is_arm) {
|
|
handle_arm_symbol (r, symbol, info, va);
|
|
}
|
|
select_flag_space (r, symbol);
|
|
/* If that's a Classed symbol (method or so) */
|
|
if (sn.classname) {
|
|
RFlagItem *fi = r_flag_get (r->flags, sn.methflag);
|
|
if (r->bin->prefix) {
|
|
char *prname = r_str_newf ("%s.%s", r->bin->prefix, sn.methflag);
|
|
free (sn.methflag);
|
|
sn.methflag = prname;
|
|
r_name_filter (sn.methflag, -1);
|
|
}
|
|
if (fi) {
|
|
r_flag_item_set_realname (fi, sn.methname);
|
|
if ((fi->offset - r->flags->base) == addr) {
|
|
// char *comment = fi->comment ? strdup (fi->comment) : NULL;
|
|
r_flag_unset (r->flags, fi);
|
|
}
|
|
} else {
|
|
fi = r_flag_set (r->flags, sn.methflag, addr, symbol->size);
|
|
char *comment = (fi && fi->comment) ? strdup (fi->comment) : NULL;
|
|
if (comment) {
|
|
r_flag_item_set_comment (fi, comment);
|
|
R_FREE (comment);
|
|
}
|
|
}
|
|
} else {
|
|
const char *n = sn.demname ? sn.demname : symbol->name;
|
|
const char *fn = sn.demflag ? sn.demflag : sn.nameflag;
|
|
char *fnp = (r->bin->prefix) ?
|
|
r_str_newf ("%s.%s", r->bin->prefix, fn):
|
|
strdup (r_str_get (fn));
|
|
RFlagItem *fi = r_flag_set (r->flags, fnp, addr, symbol->size);
|
|
if (fi) {
|
|
r_flag_item_set_realname (fi, n);
|
|
fi->demangled = (bool)(size_t)sn.demname;
|
|
} else {
|
|
if (fn) {
|
|
eprintf ("[Warning] Can't find flag (%s)\n", fn);
|
|
}
|
|
}
|
|
free (fnp);
|
|
}
|
|
if (sn.demname) {
|
|
ut64 size = symbol->size? symbol->size: 1;
|
|
r_meta_set (r->anal, R_META_TYPE_COMMENT,
|
|
addr, size, sn.demname);
|
|
}
|
|
r_flag_space_pop (r->flags);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
none = false;
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", r_symbol_name);
|
|
if (sn.demname) {
|
|
pj_ks (pj, "demname", sn.demname);
|
|
}
|
|
pj_ks (pj, "flagname", sn.nameflag);
|
|
pj_ks (pj, "realname", symbol->name);
|
|
pj_ki (pj, "ordinal", symbol->ordinal);
|
|
pj_ks (pj, "bind", symbol->bind);
|
|
pj_kn (pj, "size", (ut64)symbol->size);
|
|
pj_ks (pj, "type", symbol->type);
|
|
pj_kn (pj, "vaddr", addr);
|
|
pj_kn (pj, "paddr", symbol->paddr);
|
|
pj_kb (pj, "is_imported", symbol->is_imported);
|
|
pj_end (pj);
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
const char *name = sn.demname? sn.demname: r_symbol_name;
|
|
r_cons_printf ("0x%08"PFMT64x" %d %s%s%s\n",
|
|
addr, (int)symbol->size,
|
|
r_str_get (sn.libname), sn.libname ? " " : "",
|
|
name);
|
|
} else if (IS_MODE_SIMPLEST (mode)) {
|
|
const char *name = sn.demname? sn.demname: r_symbol_name;
|
|
r_cons_printf ("%s\n", name);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
/* Skip special symbols because we do not flag them and
|
|
* they shouldn't be printed in the rad format either */
|
|
if (is_special_symbol (symbol)) {
|
|
goto next;
|
|
}
|
|
RBinFile *binfile;
|
|
RBinPlugin *plugin;
|
|
const char *name = sn.demname? sn.demname: r_symbol_name;
|
|
if (!name) {
|
|
goto next;
|
|
}
|
|
if (symbol->is_imported) {
|
|
if (lastfs != 'i') {
|
|
r_cons_printf ("fs imports\n");
|
|
}
|
|
lastfs = 'i';
|
|
} else {
|
|
if (lastfs != 's') {
|
|
const char *fs = exponly? "exports": "symbols";
|
|
r_cons_printf ("fs %s\n", fs);
|
|
}
|
|
lastfs = 's';
|
|
}
|
|
if (r->bin->prefix || *name) { // we don't want unnamed symbol flags
|
|
char *flagname = construct_symbol_flagname ("sym", sn.libname, name, MAXFLAG_LEN_DEFAULT);
|
|
if (!flagname) {
|
|
goto next;
|
|
}
|
|
r_cons_printf ("\"f %s%s%s %u 0x%08" PFMT64x "\"\n",
|
|
r_str_get (r->bin->prefix), r->bin->prefix ? "." : "",
|
|
flagname, symbol->size, addr);
|
|
free (flagname);
|
|
}
|
|
binfile = r_bin_cur (r->bin);
|
|
plugin = r_bin_file_cur_plugin (binfile);
|
|
if (plugin && plugin->name) {
|
|
if (r_str_startswith (plugin->name, "pe")) {
|
|
char *module = strdup (r_symbol_name);
|
|
char *p = strstr (module, ".dll_");
|
|
if (p && symbol->is_imported) {
|
|
char *symname = __filterShell (p + 5);
|
|
char *m = __filterShell (module);
|
|
*p = 0;
|
|
if (r->bin->prefix) {
|
|
r_cons_printf ("\"k bin/pe/%s/%d=%s.%s\"\n",
|
|
module, symbol->ordinal, r->bin->prefix, symname);
|
|
} else {
|
|
r_cons_printf ("\"k bin/pe/%s/%d=%s\"\n",
|
|
module, symbol->ordinal, symname);
|
|
}
|
|
free (symname);
|
|
free (m);
|
|
}
|
|
free (module);
|
|
}
|
|
}
|
|
} else {
|
|
const char *bind = r_str_get_fail (symbol->bind, "NONE");
|
|
const char *type = r_str_get_fail (symbol->type, "NONE");
|
|
const char *name = r_str_getf (sn.demname? sn.demname: sn.name);
|
|
// const char *fwd = r_str_getf (symbol->forwarder);
|
|
r_table_add_rowf (table, "dXXssdss",
|
|
symbol->ordinal,
|
|
symbol->paddr,
|
|
addr,
|
|
bind,
|
|
type,
|
|
symbol->size,
|
|
r_str_get (symbol->libname),
|
|
name);
|
|
}
|
|
next:
|
|
snFini (&sn);
|
|
i++;
|
|
free (r_symbol_name);
|
|
if (exponly && firstexp) {
|
|
firstexp = false;
|
|
}
|
|
if (printHere) {
|
|
break;
|
|
}
|
|
}
|
|
if (IS_MODE_NORMAL (mode)){
|
|
if (r->table_query) {
|
|
r_table_query (table, r->table_query);
|
|
}
|
|
char *s = r_table_tostring (table);
|
|
r_cons_printf ("\n%s", s);
|
|
free (s);
|
|
}
|
|
|
|
//handle thumb and arm for entry point since they are not present in symbols
|
|
if (is_arm) {
|
|
r_list_foreach (entries, iter, entry) {
|
|
if (IS_MODE_SET (mode)) {
|
|
handle_arm_entry (r, entry, info, va);
|
|
}
|
|
}
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
if (!printHere) {
|
|
pj_end (pj);
|
|
} else if (none) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
|
|
r_spaces_pop (&r->anal->meta_spaces);
|
|
r_table_free (table);
|
|
return true;
|
|
}
|
|
|
|
static char *build_hash_string(PJ *pj, int mode, const char *chksum, ut8 *data, ut32 datalen) {
|
|
char *chkstr = NULL, *aux = NULL, *ret = NULL;
|
|
RList *hashlist = r_str_split_duplist (chksum, ",", true);
|
|
RListIter *iter;
|
|
char *hashname;
|
|
r_list_foreach (hashlist, iter, hashname) {
|
|
chkstr = r_hash_to_string (NULL, hashname, data, datalen);
|
|
if (!chkstr) {
|
|
continue;
|
|
}
|
|
if (IS_MODE_SIMPLE (mode) || IS_MODE_NORMAL (mode)) {
|
|
aux = r_str_newf (iter->n? "%s " : "%s", chkstr);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_ks (pj, hashname, chkstr);
|
|
} else {
|
|
aux = r_str_newf ("%s=%s ", hashname, chkstr);
|
|
}
|
|
ret = r_str_append (ret, aux);
|
|
free (chkstr);
|
|
free (aux);
|
|
}
|
|
r_list_free (hashlist);
|
|
return ret;
|
|
}
|
|
|
|
static char *filter_hash_string(const char *chksum) {
|
|
if (!chksum) {
|
|
return NULL;
|
|
}
|
|
|
|
char *aux, *ret = NULL;
|
|
bool isFirst = true;
|
|
RList *hashlist = r_str_split_duplist (chksum, ",", true);
|
|
RListIter *iter;
|
|
char *hashname;
|
|
r_list_foreach (hashlist, iter, hashname) {
|
|
if (r_hash_name_to_bits (hashname)) {
|
|
aux = r_str_newf (isFirst? "%s" : ", %s", hashname);
|
|
ret = r_str_append (ret, aux);
|
|
free (aux);
|
|
if (isFirst) {
|
|
isFirst = false;
|
|
}
|
|
}
|
|
}
|
|
r_list_free (hashlist);
|
|
return ret;
|
|
}
|
|
|
|
typedef struct {
|
|
const char *uri;
|
|
int perm;
|
|
RIODesc *desc;
|
|
} FindFile;
|
|
|
|
static bool findFile(void *user, void *data, ut32 id) {
|
|
FindFile *res = (FindFile*)user;
|
|
RIODesc *desc = (RIODesc*)data;
|
|
if (desc->perm && res->perm && !strcmp (desc->uri, res->uri)) {
|
|
res->desc = desc;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static RIODesc *findReusableFile(RIO *io, const char *uri, int perm) {
|
|
FindFile arg = {
|
|
.uri = uri,
|
|
.perm = perm,
|
|
.desc = NULL,
|
|
};
|
|
r_id_storage_foreach (io->files, findFile, &arg);
|
|
return arg.desc;
|
|
}
|
|
|
|
static bool io_create_mem_map(RIO *io, RBinSection *sec, ut64 at) {
|
|
r_return_val_if_fail (io && sec, false);
|
|
|
|
bool reused = false;
|
|
ut64 gap = sec->vsize - sec->size;
|
|
char *uri = r_str_newf ("null://%"PFMT64u, gap);
|
|
RIODesc *desc = findReusableFile (io, uri, sec->perm);
|
|
if (desc) {
|
|
RIOMap *map = r_io_map_get_at (io, at);
|
|
if (!map) {
|
|
r_io_map_add (io, desc->fd, desc->perm, 0LL, at, gap);
|
|
}
|
|
reused = true;
|
|
}
|
|
if (!desc) {
|
|
desc = r_io_open_at (io, uri, sec->perm, 0664, at);
|
|
}
|
|
free (uri);
|
|
if (!desc) {
|
|
return false;
|
|
}
|
|
// this works, because new maps are always born on the top
|
|
RIOMap *map = r_io_map_get_at (io, at);
|
|
// check if the mapping failed
|
|
if (!map) {
|
|
if (!reused) {
|
|
r_io_desc_close (desc);
|
|
}
|
|
return false;
|
|
}
|
|
// let the section refere to the map as a memory-map
|
|
free (map->name);
|
|
map->name = r_str_newf ("mmap.%s", sec->name);
|
|
return true;
|
|
}
|
|
|
|
static void add_section(RCore *core, RBinSection *sec, ut64 addr, int fd) {
|
|
if (!r_io_desc_get (core->io, fd) || UT64_ADD_OVFCHK (sec->size, sec->paddr) ||
|
|
UT64_ADD_OVFCHK (sec->size, addr) || !sec->vsize) {
|
|
return;
|
|
}
|
|
|
|
ut64 size = sec->vsize;
|
|
// if there is some part of the section that needs to be zeroed by the loader
|
|
// we add a null map that takes care of it
|
|
if (sec->vsize > sec->size) {
|
|
if (!io_create_mem_map (core->io, sec, addr + sec->size)) {
|
|
return;
|
|
}
|
|
size = sec->size;
|
|
if (!size) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// then we map the part of the section that comes from the physical file
|
|
char *map_name = r_str_newf ("fmap.%s", sec->name);
|
|
if (!map_name) {
|
|
return;
|
|
}
|
|
r_name_filter (map_name, R_FLAG_NAME_SIZE);
|
|
int perm = sec->perm;
|
|
// workaround to force exec bit in text section
|
|
if (sec->name && strstr (sec->name, "text")) {
|
|
perm |= R_PERM_X;
|
|
}
|
|
|
|
RIOMap *map = r_io_map_add (core->io, fd, perm, sec->paddr, addr, size);
|
|
if (!map) {
|
|
free (map_name);
|
|
return;
|
|
}
|
|
map->name = map_name;
|
|
return;
|
|
}
|
|
|
|
struct io_bin_section_info_t {
|
|
RBinSection *sec;
|
|
ut64 addr;
|
|
int fd;
|
|
};
|
|
|
|
/* Map Sections to Segments https://github.com/radareorg/radare2/issues/14647 */
|
|
static int bin_map_sections_to_segments(RBin *bin, PJ *pj, int mode) {
|
|
RListIter *iter, *iter2;
|
|
RBinSection *section = NULL, *segment = NULL;
|
|
RList *sections = r_list_new ();
|
|
RList *segments = r_list_new ();
|
|
RList *tmp = r_bin_get_sections (bin);
|
|
RTable *table = r_table_new ("segments");
|
|
RTableColumnType *typeString = r_table_type ("string");
|
|
|
|
r_table_add_column (table, typeString, "Segment", 0);
|
|
r_table_add_column (table, typeString, "Section", 0);
|
|
|
|
r_list_foreach (tmp, iter, section) {
|
|
RList *list = section->is_segment? segments: sections;
|
|
r_list_append (list, section);
|
|
}
|
|
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
pj_o (pj);
|
|
}
|
|
|
|
r_list_foreach (segments, iter, segment) {
|
|
RInterval segment_itv = (RInterval){segment->vaddr, segment->size};
|
|
char *tmp2 = r_str_new ("");
|
|
r_list_foreach (sections, iter2, section) {
|
|
RInterval section_itv = (RInterval){section->vaddr, section->size};
|
|
if (r_itv_begin (section_itv) >= r_itv_begin (segment_itv) && r_itv_end (section_itv) <= r_itv_end (segment_itv) && section->name[0]) {
|
|
tmp2 = r_str_appendf (tmp2, "%s ", section->name);
|
|
}
|
|
}
|
|
r_table_add_row (table, segment->name, tmp2, 0);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ks (pj, segment->name, tmp2);
|
|
}
|
|
free (tmp2);
|
|
}
|
|
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
}
|
|
|
|
if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_printf ("Section to Segment mapping:\n");
|
|
char *s = r_table_tostring (table);
|
|
r_cons_printf ("%s\n", s);
|
|
free (s);
|
|
}
|
|
r_list_free (segments);
|
|
r_table_free (table);
|
|
return true;
|
|
}
|
|
|
|
static int bin_sections(RCore *r, PJ *pj, int mode, ut64 laddr, int va, ut64 at, const char *name, const char *chksum, bool print_segments) {
|
|
char *str = NULL;
|
|
RBinSection *section;
|
|
RBinInfo *info = NULL;
|
|
RList *sections;
|
|
RListIter *iter;
|
|
RTable *table = r_core_table (r, "sections");
|
|
r_return_val_if_fail (table, false);
|
|
int i = 0;
|
|
int fd = -1;
|
|
bool printHere = false;
|
|
sections = r_bin_get_sections (r->bin);
|
|
#if LOAD_BSS_MALLOC
|
|
const bool inDebugger = r_config_get_b (r->config, "cfg.debug");
|
|
#endif
|
|
HtPP *dup_chk_ht = ht_pp_new0 ();
|
|
bool ret = false;
|
|
const char *type = print_segments ? "segment" : "section";
|
|
bool segments_only = true;
|
|
RList *io_section_info = NULL;
|
|
ut64 bin_hashlimit = r_config_get_i (r->config, "bin.hashlimit");
|
|
ut64 filesize = (r->io->desc) ? r_io_fd_size (r->io, r->io->desc->fd): 0;
|
|
|
|
if (!dup_chk_ht) {
|
|
return false;
|
|
}
|
|
|
|
if (chksum && *chksum == '.') {
|
|
printHere = true;
|
|
chksum++;
|
|
}
|
|
char *hashtypes = filter_hash_string (chksum);
|
|
if (IS_MODE_EQUAL (mode)) {
|
|
int cols = r_cons_get_size (NULL);
|
|
RList *list = r_list_newf ((RListFree) r_listinfo_free);
|
|
if (!list) {
|
|
free (hashtypes);
|
|
return false;
|
|
}
|
|
RBinSection *s;
|
|
r_list_foreach (sections, iter, s) {
|
|
char humansz[8];
|
|
if (print_segments != s->is_segment) {
|
|
continue;
|
|
}
|
|
RInterval pitv = (RInterval){s->paddr, s->size};
|
|
RInterval vitv = (RInterval){s->vaddr, s->vsize};
|
|
r_num_units (humansz, sizeof (humansz), s->size);
|
|
RListInfo *info = r_listinfo_new (s->name, pitv, vitv, s->perm, strdup (humansz));
|
|
r_list_append (list, info);
|
|
}
|
|
RTable *table = r_core_table (r, "sections");
|
|
r_table_visual_list (table, list, r->offset, -1, cols, r->io->va);
|
|
if (r->table_query) {
|
|
r_table_query (table, r->table_query);
|
|
}
|
|
{
|
|
char *s = r_table_tostring (table);
|
|
r_cons_printf ("\n%s\n", s);
|
|
free (s);
|
|
}
|
|
r_table_free (table);
|
|
r_list_free (list);
|
|
goto out;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
if (!printHere) {
|
|
pj_a (pj);
|
|
}
|
|
} else if (IS_MODE_RAD (mode) && at == UT64_MAX) {
|
|
r_cons_printf ("fs %ss\n", type);
|
|
} else if (IS_MODE_NORMAL (mode) && at == UT64_MAX && !printHere) {
|
|
r_cons_printf ("[%s]\n", print_segments ? "Segments" : "Sections");
|
|
} else if (IS_MODE_NORMAL (mode) && printHere) {
|
|
r_cons_printf ("Current section\n");
|
|
} else if (IS_MODE_SET (mode)) {
|
|
if (r && r->io->desc) {
|
|
fd = r->io->desc->fd;
|
|
}
|
|
r_flag_space_set (r->flags, print_segments? R_FLAGS_FS_SEGMENTS: R_FLAGS_FS_SECTIONS);
|
|
}
|
|
if (IS_MODE_NORMAL (mode)) {
|
|
if (hashtypes) {
|
|
r_table_set_columnsf (table, "dXxXxsss",
|
|
"nth", "paddr", "size", "vaddr", "vsize", "perm", hashtypes, "name");
|
|
} else {
|
|
r_table_set_columnsf (table, "dXxXxss",
|
|
"nth", "paddr", "size", "vaddr", "vsize", "perm", "name");
|
|
}
|
|
// r_table_align (table, 0, R_TABLE_ALIGN_CENTER);
|
|
r_table_align (table, 2, R_TABLE_ALIGN_RIGHT);
|
|
r_table_align (table, 4, R_TABLE_ALIGN_RIGHT);
|
|
}
|
|
if (IS_MODE_SET (mode)) {
|
|
r_list_foreach (sections, iter, section) {
|
|
if (!section->is_segment) {
|
|
segments_only = false;
|
|
break;
|
|
}
|
|
}
|
|
io_section_info = r_list_newf ((RListFree)free);
|
|
}
|
|
r_list_foreach (sections, iter, section) {
|
|
char perms[] = "----";
|
|
int va_sect = va;
|
|
ut64 addr;
|
|
|
|
if (va && !(section->perm & R_PERM_R)) {
|
|
va_sect = VA_NOREBASE;
|
|
}
|
|
addr = rva (r->bin, section->paddr, section->vaddr, va_sect);
|
|
|
|
if (name && strcmp (section->name, name)) {
|
|
continue;
|
|
}
|
|
|
|
if ((printHere && !(section->paddr <= r->offset && r->offset < (section->paddr + section->size)))
|
|
&& (printHere && !(addr <= r->offset && r->offset < (addr + section->size)))) {
|
|
continue;
|
|
}
|
|
|
|
if (at != UT64_MAX && (!section->size || !is_in_range (at, addr, section->size))) {
|
|
continue;
|
|
}
|
|
|
|
if (section->is_segment != print_segments) {
|
|
continue;
|
|
}
|
|
// XXX use r_str_perm instead of doing it here imho
|
|
if (section->perm & R_PERM_SHAR) {
|
|
perms[0] = 's';
|
|
}
|
|
if (section->perm & R_PERM_R) {
|
|
perms[1] = 'r';
|
|
}
|
|
if (section->perm & R_PERM_W) {
|
|
perms[2] = 'w';
|
|
}
|
|
if (section->perm & R_PERM_X) {
|
|
perms[3] = 'x';
|
|
}
|
|
const char *arch = NULL;
|
|
int bits = 0;
|
|
if (section->arch || section->bits) {
|
|
arch = section->arch;
|
|
bits = section->bits;
|
|
}
|
|
if (info) {
|
|
if (!arch) {
|
|
arch = info->arch;
|
|
}
|
|
if (!bits) {
|
|
bits = info->bits;
|
|
}
|
|
}
|
|
if (!arch) {
|
|
arch = r_config_get (r->config, "asm.arch");
|
|
}
|
|
if (!bits) {
|
|
bits = R_SYS_BITS;
|
|
}
|
|
if (IS_MODE_RAD (mode)) {
|
|
char *fname = r_str_newf ("%s.%s", type, section->name);
|
|
r_name_filter (fname, -1);
|
|
r_cons_printf ("\"f %s 1 0x%08"PFMT64x"\"\n", fname, section->vaddr);
|
|
free (fname);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
#if LOAD_BSS_MALLOC
|
|
if (!strcmp (section->name, ".bss")) {
|
|
// check if there's already a file opened there
|
|
int loaded = 0;
|
|
RListIter *iter;
|
|
RIOMap *m;
|
|
r_list_foreach (r->io->maps, iter, m) {
|
|
if (m->from == addr) {
|
|
loaded = 1;
|
|
}
|
|
}
|
|
if (!loaded && !inDebugger) {
|
|
r_core_cmdf (r, "on malloc://%d 0x%"PFMT64x" # bss\n",
|
|
section->vsize, addr);
|
|
}
|
|
}
|
|
#endif
|
|
if (section->format) {
|
|
// This is damn slow if section vsize is HUGE
|
|
if (section->vsize < 1024 * 1024 * 2) {
|
|
r_core_cmdf (r, "%s @ 0x%" PFMT64x,
|
|
section->format, section->vaddr);
|
|
}
|
|
}
|
|
if (r->bin->prefix) {
|
|
str = r_str_newf ("%s.%s.%s", r->bin->prefix, type, name);
|
|
} else {
|
|
str = r_str_newf ("%s.%s", type, section->name);
|
|
}
|
|
r_name_filter (str, R_FLAG_NAME_SIZE);
|
|
ut64 size = r->io->va? section->vsize: section->size;
|
|
r_flag_set (r->flags, str, addr, size);
|
|
R_FREE (str);
|
|
|
|
if (!section->is_segment || segments_only) {
|
|
char *pfx = r->bin->prefix;
|
|
str = r_str_newf ("[%02d] %s %s size %" PFMT64d " named %s%s%s",
|
|
i, perms, type, size,
|
|
r_str_get (pfx), pfx? ".": "", section->name);
|
|
r_meta_set (r->anal, R_META_TYPE_COMMENT, addr, 1, str);
|
|
R_FREE (str);
|
|
}
|
|
if (section->add) {
|
|
bool found;
|
|
str = r_str_newf ("%" PFMT64x ".%" PFMT64x ".%" PFMT64x ".%" PFMT64x
|
|
".%" PFMT32u ".%s.%" PFMT32u ".%d",
|
|
section->paddr, addr, section->size,
|
|
section->vsize, section->perm, section->name,
|
|
r->bin->cur->id, fd);
|
|
ht_pp_find (dup_chk_ht, str, &found);
|
|
if (!found) {
|
|
// can't directly add maps because they
|
|
// need to be reversed, otherwise for
|
|
// the way IO works maps would be shown
|
|
// in reverse order
|
|
struct io_bin_section_info_t *ibs = R_NEW (struct io_bin_section_info_t);
|
|
if (!ibs) {
|
|
eprintf ("Could not allocate memory\n");
|
|
goto out;
|
|
}
|
|
ibs->sec = section;
|
|
ibs->addr = addr;
|
|
ibs->fd = fd;
|
|
r_list_append (io_section_info, ibs);
|
|
ht_pp_insert (dup_chk_ht, str, NULL);
|
|
}
|
|
R_FREE (str);
|
|
}
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
char *hashstr = NULL;
|
|
if (hashtypes) {
|
|
int datalen = section->size;
|
|
int limit = filesize? R_MIN (filesize, bin_hashlimit): bin_hashlimit;
|
|
if (datalen > 0 && datalen < limit) {
|
|
ut8 *data = malloc (datalen);
|
|
if (!data) {
|
|
goto out;
|
|
}
|
|
int dl = r_io_pread_at (r->io, section->paddr, data, datalen);
|
|
if (dl == datalen) {
|
|
hashstr = build_hash_string (pj, mode, hashtypes, data, datalen);
|
|
} else if (r->bin->verbose) {
|
|
eprintf ("Cannot read section at 0x%08"PFMT64x"\n", section->paddr);
|
|
}
|
|
free (data);
|
|
} else if (r->bin->verbose) {
|
|
eprintf ("Section at 0x%08"PFMT64x" larger than bin.hashlimit\n", section->paddr);
|
|
}
|
|
}
|
|
r_cons_printf ("0x%"PFMT64x" 0x%"PFMT64x" %s %s%s%s\n",
|
|
addr, addr + section->size,
|
|
perms,
|
|
r_str_get (hashstr), hashstr ? " " : "",
|
|
section->name
|
|
);
|
|
free (hashstr);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", section->name);
|
|
pj_kN (pj, "size", section->size);
|
|
pj_kN (pj, "vsize", section->vsize);
|
|
pj_ks (pj, "perm", perms);
|
|
if (hashtypes && (int)section->size > 0) {
|
|
int datalen = section->size;
|
|
int limit = filesize? R_MIN (filesize, bin_hashlimit): bin_hashlimit;
|
|
if (datalen > 0 && datalen < limit) {
|
|
ut8 *data = malloc (datalen);
|
|
if (!data) {
|
|
goto out;
|
|
}
|
|
int dl = r_io_pread_at (r->io, section->paddr, data, datalen);
|
|
if (dl == datalen) {
|
|
free (build_hash_string (pj, mode, hashtypes, data, datalen));
|
|
} else if (r->bin->verbose) {
|
|
eprintf ("Cannot read section at 0x%08"PFMT64x"\n", section->paddr);
|
|
}
|
|
free (data);
|
|
} else if (r->bin->verbose) {
|
|
eprintf ("Section at 0x%08"PFMT64x" larger than bin.hashlimit\n", section->paddr);
|
|
}
|
|
}
|
|
pj_kN (pj, "paddr", section->paddr);
|
|
pj_kN (pj, "vaddr", addr);
|
|
pj_end (pj);
|
|
} else {
|
|
char *hashstr = NULL, str[128];
|
|
if (hashtypes && section->size > 0) {
|
|
int datalen = section->size;
|
|
int limit = filesize? R_MIN (filesize, bin_hashlimit): bin_hashlimit;
|
|
if (datalen > 0 && datalen < limit) {
|
|
ut8 *data = malloc (datalen);
|
|
if (!data) {
|
|
goto out;
|
|
}
|
|
int dl = r_io_pread_at (r->io, section->paddr, data, datalen);
|
|
if (dl == datalen) {
|
|
hashstr = build_hash_string (pj, mode, hashtypes, data, datalen);
|
|
} else if (r->bin->verbose) {
|
|
eprintf ("Cannot read section at 0x%08"PFMT64x"\n", section->paddr);
|
|
}
|
|
free (data);
|
|
} else if (r->bin->verbose) {
|
|
eprintf ("Section at 0x%08"PFMT64x" larger than bin.hashlimit\n", section->paddr);
|
|
}
|
|
}
|
|
if (section->arch || section->bits) {
|
|
snprintf (str, sizeof (str), "arch=%s bits=%d ",
|
|
r_str_get (arch), bits);
|
|
} else {
|
|
str[0] = 0;
|
|
}
|
|
r_strf_buffer (128);
|
|
const char *section_name = (r->bin->prefix)
|
|
? r_strf ("%s.%s", r->bin->prefix, section->name)
|
|
: section->name;
|
|
// seems like asm.bits is a bitmask that seems to be always 32,64
|
|
// const char *asmbits = r_str_sysbits (bits);
|
|
if (hashtypes) {
|
|
r_table_add_rowf (table, "dXxXxsss", i,
|
|
(ut64)section->paddr, (ut64)section->size,
|
|
(ut64)addr, (ut64)section->vsize,
|
|
perms, r_str_get (hashstr), section_name);
|
|
} else {
|
|
r_table_add_rowf (table, "dXxXxss", i,
|
|
(ut64)section->paddr, (ut64)section->size,
|
|
(ut64)addr, (ut64)section->vsize,
|
|
perms, section_name);
|
|
}
|
|
free (hashstr);
|
|
}
|
|
i++;
|
|
if (printHere) {
|
|
break;
|
|
}
|
|
}
|
|
if (IS_MODE_SET (mode) && !r_io_desc_is_dbg (r->io->desc)) {
|
|
RListIter *it;
|
|
struct io_bin_section_info_t *ibs;
|
|
r_list_foreach_prev (io_section_info, it, ibs) {
|
|
add_section (r, ibs->sec, ibs->addr, ibs->fd);
|
|
}
|
|
}
|
|
r_list_free (io_section_info);
|
|
io_section_info = NULL;
|
|
if (IS_MODE_JSON (mode) && !printHere) {
|
|
pj_end (pj);
|
|
} else if (IS_MODE_NORMAL (mode) && at == UT64_MAX && !printHere) {
|
|
// r_cons_printf ("\n%i sections\n", i);
|
|
}
|
|
|
|
ret = true;
|
|
out:
|
|
if (IS_MODE_NORMAL (mode)) {
|
|
if (r->table_query) {
|
|
r_table_query (table, r->table_query);
|
|
}
|
|
char *s = r_table_tostring (table);
|
|
r_cons_printf ("\n%s\n", s);
|
|
free (s);
|
|
}
|
|
free (hashtypes);
|
|
r_table_free (table);
|
|
ht_pp_free (dup_chk_ht);
|
|
return ret;
|
|
}
|
|
|
|
static int bin_fields(RCore *r, PJ *pj, int mode, int va) {
|
|
RList *fields;
|
|
RListIter *iter;
|
|
RBinField *field;
|
|
int i = 0;
|
|
RBin *bin = r->bin;
|
|
|
|
if (!(fields = r_bin_get_fields (bin))) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
}
|
|
return false;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_println ("fs header");
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_println ("[Header fields]");
|
|
}
|
|
r_list_foreach (fields, iter, field) {
|
|
ut64 addr = rva (bin, field->paddr, field->vaddr, va);
|
|
|
|
if (IS_MODE_RAD (mode)) {
|
|
char *n = __filterQuotedShell (field->name);
|
|
r_name_filter (n, -1);
|
|
r_cons_printf ("\"f header.%s 1 0x%08"PFMT64x"\"\n", n, addr);
|
|
if (field->comment && *field->comment) {
|
|
char *e = sdb_encode ((const ut8*)field->comment, -1);
|
|
r_cons_printf ("CCu %s @ 0x%"PFMT64x"\n", e, addr);
|
|
free (e);
|
|
char *f = __filterShell (field->format);
|
|
r_cons_printf ("Cf %d %s @ 0x%"PFMT64x"\n", field->size, f, addr);
|
|
free (f);
|
|
}
|
|
if (field->format && *field->format && !field->format_named) {
|
|
r_cons_printf ("pf.%s %s\n", n, field->format);
|
|
}
|
|
free (n);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", field->name);
|
|
pj_kN (pj, "vaddr", field->vaddr);
|
|
pj_kN (pj, "paddr", field->paddr);
|
|
if (field->comment && *field->comment) {
|
|
// TODO: filter comment before json
|
|
pj_ks (pj, "comment", field->comment);
|
|
}
|
|
if (field->format && *field->format) {
|
|
// TODO: filter comment before json
|
|
pj_ks (pj, "format", field->format);
|
|
}
|
|
char *o = r_core_cmd_strf (r, "pfj%c%s@0x%"PFMT64x,
|
|
field->format_named ? '.' : ' ', field->format, field->vaddr);
|
|
if (o && *o) {
|
|
r_str_trim_tail(o);
|
|
pj_k (pj, "pf");
|
|
pj_j (pj, o);
|
|
}
|
|
free (o);
|
|
pj_end (pj);
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
const bool haveComment = (field->comment && *field->comment);
|
|
r_cons_printf ("0x%08"PFMT64x" 0x%08"PFMT64x" %s%s%s\n",
|
|
field->vaddr, field->paddr, field->name,
|
|
haveComment? "; ": "",
|
|
haveComment? field->comment: "");
|
|
}
|
|
i++;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_printf ("\n%i fields\n", i);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static char *get_rp(const char *rtype) {
|
|
char *rp = NULL;
|
|
switch (rtype[0]) {
|
|
case 'v':
|
|
rp = strdup ("void");
|
|
break;
|
|
case 'c':
|
|
rp = strdup ("char");
|
|
break;
|
|
case 'i':
|
|
rp = strdup ("int");
|
|
break;
|
|
case 's':
|
|
rp = strdup ("short");
|
|
break;
|
|
case 'l':
|
|
rp = strdup ("long");
|
|
break;
|
|
case 'q':
|
|
rp = strdup ("long long");
|
|
break;
|
|
case 'C':
|
|
rp = strdup ("unsigned char");
|
|
break;
|
|
case 'I':
|
|
rp = strdup ("unsigned int");
|
|
break;
|
|
case 'S':
|
|
rp = strdup ("unsigned short");
|
|
break;
|
|
case 'L':
|
|
rp = strdup ("unsigned long");
|
|
break;
|
|
case 'Q':
|
|
rp = strdup ("unsigned long long");
|
|
break;
|
|
case 'f':
|
|
rp = strdup ("float");
|
|
break;
|
|
case 'd':
|
|
rp = strdup ("double");
|
|
break;
|
|
case 'D':
|
|
rp = strdup ("long double");
|
|
break;
|
|
case 'B':
|
|
rp = strdup ("bool");
|
|
break;
|
|
case '#':
|
|
rp = strdup ("CLASS");
|
|
break;
|
|
default:
|
|
rp = strdup ("unknown");
|
|
break;
|
|
}
|
|
return rp;
|
|
}
|
|
|
|
static int bin_trycatch(RCore *core, PJ *pj, int mode) {
|
|
RBinFile *bf = r_bin_cur (core->bin);
|
|
RListIter *iter;
|
|
RBinTrycatch *tc;
|
|
RList *trycatch = r_bin_file_get_trycatch (bf);
|
|
int idx = 0;
|
|
// fixme: json mode
|
|
r_list_foreach (trycatch, iter, tc) {
|
|
r_cons_printf ("f try.%d.%"PFMT64x".from=0x%08"PFMT64x"\n", idx, tc->source, tc->from);
|
|
r_cons_printf ("f try.%d.%"PFMT64x".to=0x%08"PFMT64x"\n", idx, tc->source, tc->to);
|
|
r_cons_printf ("f try.%d.%"PFMT64x".catch=0x%08"PFMT64x"\n", idx, tc->source, tc->handler);
|
|
idx++;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// https://nshipster.com/type-encodings/
|
|
static char *objc_type_toc(const char *objc_type) {
|
|
if (!objc_type) {
|
|
return strdup ("void*");
|
|
}
|
|
if (*objc_type == '^' && objc_type[1] == '{') {
|
|
char *a = strdup (objc_type + 2);
|
|
char *b = strchr (a, '>');
|
|
if (b) {
|
|
*b = 0;
|
|
}
|
|
a[strlen (a) - 1] = 0;
|
|
return a;
|
|
}
|
|
if (*objc_type == '<') {
|
|
char *a = strdup (objc_type + 1);
|
|
char *b = strchr (a, '>');
|
|
if (b) {
|
|
*b = 0;
|
|
}
|
|
return a;
|
|
}
|
|
if (!strcmp (objc_type, "f")) { return strdup ("float"); }
|
|
if (!strcmp (objc_type, "d")) { return strdup ("double"); }
|
|
if (!strcmp (objc_type, "i")) { return strdup ("int"); }
|
|
if (!strcmp (objc_type, "s")) { return strdup ("short"); }
|
|
if (!strcmp (objc_type, "l")) { return strdup ("long"); }
|
|
if (!strcmp (objc_type, "L")) { return strdup ("unsigned long"); }
|
|
if (!strcmp (objc_type, "*")) { return strdup ("char*"); }
|
|
if (!strcmp (objc_type, "c")) { return strdup ("bool"); }
|
|
if (!strcmp (objc_type, "v")) { return strdup ("void"); }
|
|
if (!strcmp (objc_type, "#")) { return strdup ("class"); }
|
|
if (!strcmp (objc_type, "B")) { return strdup ("cxxbool"); }
|
|
if (!strcmp (objc_type, "Q")) { return strdup ("uint64_t"); }
|
|
if (!strcmp (objc_type, "q")) { return strdup ("long long"); }
|
|
if (!strcmp (objc_type, "C")) { return strdup ("uint8_t"); }
|
|
if (strlen (objc_type) == 1) {
|
|
eprintf ("Unknown objc type '%s'\n", objc_type);
|
|
}
|
|
if (r_str_startswith (objc_type, "@\"")) {
|
|
char *s = r_str_newf ("struct %s", objc_type + 2);
|
|
s[strlen (s) - 1] = '*';
|
|
return s;
|
|
}
|
|
return strdup (objc_type);
|
|
}
|
|
|
|
static char *objc_name_toc(const char *objc_name) {
|
|
const char *n = r_str_lchr (objc_name, ')');
|
|
char *s = strdup (n? n + 1: objc_name);
|
|
char *p = strchr (s, '(');
|
|
if (p) {
|
|
*p = 0;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
static void classdump_c(RCore *r, RBinClass *c) {
|
|
r_cons_printf ("typedef struct class_%s {\n", c->name);
|
|
RListIter *iter2;
|
|
RBinField *f;
|
|
r_list_foreach (c->fields, iter2, f) {
|
|
if (f->type && f->name) {
|
|
char *n = objc_name_toc (f->name);
|
|
char *t = objc_type_toc (f->type);
|
|
r_cons_printf (" %s %s; // %d\n", t, n, f->offset);
|
|
free (t);
|
|
free (n);
|
|
}
|
|
}
|
|
r_cons_printf ("} %s;\n", c->name);
|
|
}
|
|
|
|
static void classdump_objc(RCore *r, RBinClass *c) {
|
|
if (c->super) {
|
|
r_cons_printf ("@interface %s : %s\n{\n", c->name, c->super);
|
|
} else {
|
|
r_cons_printf ("@interface %s\n{\n", c->name);
|
|
}
|
|
RListIter *iter2, *iter3;
|
|
RBinField *f;
|
|
RBinSymbol *sym;
|
|
r_list_foreach (c->fields, iter2, f) {
|
|
if (f->name && r_regex_match ("ivar","e", f->name)) {
|
|
r_cons_printf (" %s %s\n", f->type, f->name);
|
|
}
|
|
}
|
|
r_cons_printf ("}\n");
|
|
r_list_foreach (c->methods, iter3, sym) {
|
|
if (sym->rtype && sym->rtype[0] != '@') {
|
|
char *rp = get_rp (sym->rtype);
|
|
r_cons_printf ("%s (%s) %s\n",
|
|
strncmp (sym->type, R_BIN_TYPE_METH_STR, 4)? "+": "-",
|
|
rp, sym->dname? sym->dname: sym->name);
|
|
free (rp);
|
|
} else if (sym->type) {
|
|
r_cons_printf ("%s (id) %s\n",
|
|
strncmp (sym->type, R_BIN_TYPE_METH_STR, 4)? "+": "-",
|
|
sym->dname? sym->dname: sym->name);
|
|
}
|
|
}
|
|
r_cons_printf ("@end\n");
|
|
}
|
|
|
|
static void classdump_java(RCore *r, RBinClass *c) {
|
|
RBinField *f;
|
|
RListIter *iter2, *iter3;
|
|
RBinSymbol *sym;
|
|
char *pn = strdup (c->name);
|
|
char *cn = (char *)r_str_rchr (pn, NULL, '/');
|
|
if (cn) {
|
|
*cn = 0;
|
|
cn++;
|
|
r_str_replace_char (pn, '/', '.');
|
|
}
|
|
r_cons_printf ("package %s;\n\n", pn);
|
|
r_cons_printf ("public class %s {\n", cn);
|
|
free (pn);
|
|
r_list_foreach (c->fields, iter2, f) {
|
|
if (f->name && r_regex_match ("ivar", "e", f->name)) {
|
|
r_cons_printf (" public %s %s\n", f->type, f->name);
|
|
}
|
|
}
|
|
r_list_foreach (c->methods, iter3, sym) {
|
|
const char *mn = sym->dname? sym->dname: sym->name;
|
|
const char *ms = strstr (mn, "method.");
|
|
if (ms) {
|
|
mn = ms + strlen ("method.");
|
|
}
|
|
r_cons_printf (" public %s ();\n", mn);
|
|
}
|
|
r_cons_printf ("}\n\n");
|
|
}
|
|
|
|
static int bin_classes(RCore *r, PJ *pj, int mode) {
|
|
RListIter *iter, *iter2, *iter3;
|
|
RBinSymbol *sym;
|
|
RBinClass *c;
|
|
RBinField *f;
|
|
char *name;
|
|
RList *cs = r_bin_get_classes (r->bin);
|
|
if (!cs) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
pj_end (pj);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
// XXX: support for classes is broken and needs more love
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
if (!r_config_get_i (r->config, "bin.classes")) {
|
|
return false;
|
|
}
|
|
r_flag_space_set (r->flags, R_FLAGS_FS_CLASSES);
|
|
} else if (IS_MODE_RAD (mode) && !IS_MODE_CLASSDUMP (mode)) {
|
|
r_cons_println ("fs classes");
|
|
}
|
|
|
|
r_list_foreach (cs, iter, c) {
|
|
if (!c || !c->name || !c->name[0]) {
|
|
continue;
|
|
}
|
|
name = strdup (c->name);
|
|
r_name_filter (name, -1);
|
|
ut64 at_min = UT64_MAX;
|
|
ut64 at_max = 0LL;
|
|
|
|
r_list_foreach (c->methods, iter2, sym) {
|
|
if (sym->vaddr) {
|
|
if (sym->vaddr < at_min) {
|
|
at_min = sym->vaddr;
|
|
}
|
|
if (sym->vaddr + sym->size > at_max) {
|
|
at_max = sym->vaddr + sym->size;
|
|
}
|
|
}
|
|
}
|
|
if (at_min == UT64_MAX) {
|
|
at_min = c->addr;
|
|
at_max = c->addr; // XXX + size?
|
|
}
|
|
|
|
if (IS_MODE_SET (mode)) {
|
|
r_strf_var (classname, 256, "class.%s", name);
|
|
r_flag_set (r->flags, classname, c->addr, 1);
|
|
r_list_foreach (c->methods, iter2, sym) {
|
|
char *mflags = r_core_bin_method_flags_str (sym->method_flags, mode);
|
|
r_strf_var (method, 256, "method%s.%s.%s", mflags, c->name, sym->name);
|
|
R_FREE (mflags);
|
|
r_name_filter (method, -1);
|
|
r_flag_set (r->flags, method, sym->vaddr, 1);
|
|
}
|
|
#if 0
|
|
r_list_foreach (c->fields, iter2, f) {
|
|
char *fn = r_str_newf ("field.%s.%s", classname, f->name);
|
|
ut64 at = f->vaddr; // sym->vaddr + (f->vaddr & 0xffff);
|
|
r_flag_set (r->flags, fn, at, 1);
|
|
free (fn);
|
|
}
|
|
#endif
|
|
} else if (IS_MODE_SIMPLEST (mode)) {
|
|
r_cons_printf ("%s\n", c->name);
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("0x%08"PFMT64x" [0x%08"PFMT64x" - 0x%08"PFMT64x"] %s%s%s\n",
|
|
c->addr, at_min, at_max, c->name, c->super ? " " : "",
|
|
r_str_get (c->super));
|
|
} else if (IS_MODE_CLASSDUMP (mode)) {
|
|
if (c) {
|
|
RBinFile *bf = r_bin_cur (r->bin);
|
|
if (bf && bf->o) {
|
|
if (IS_MODE_RAD (mode)) {
|
|
classdump_c (r, c);
|
|
} else if (bf->o->lang == R_BIN_NM_JAVA || (bf->o->info && bf->o->info->lang && strstr (bf->o->info->lang, "dalvik"))) {
|
|
classdump_java (r, c);
|
|
} else {
|
|
classdump_objc (r, c);
|
|
}
|
|
} else {
|
|
classdump_objc (r, c);
|
|
}
|
|
}
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
char *n = __filterShell (name);
|
|
r_cons_printf ("\"f class.%s = 0x%"PFMT64x"\"\n", n, at_min);
|
|
free (n);
|
|
if (c->super) {
|
|
char *cn = c->name; // __filterShell (c->name);
|
|
char *su = c->super; // __filterShell (c->super);
|
|
r_cons_printf ("\"f super.%s.%s = %d\"\n",
|
|
cn, su, c->index);
|
|
// free (cn);
|
|
// free (su);
|
|
}
|
|
r_list_foreach (c->methods, iter2, sym) {
|
|
char *mflags = r_core_bin_method_flags_str (sym->method_flags, mode);
|
|
char *n = c->name; // __filterShell (c->name);
|
|
char *sn = sym->name; //__filterShell (sym->name);
|
|
char *cmd = r_str_newf ("\"f method%s.%s.%s = 0x%"PFMT64x"\"\n", mflags, n, sn, sym->vaddr);
|
|
// free (n);
|
|
// free (sn);
|
|
if (cmd) {
|
|
r_str_replace_char (cmd, ' ', '_');
|
|
if (strlen (cmd) > 2) {
|
|
cmd[2] = ' ';
|
|
}
|
|
char *eq = (char *)r_str_rchr (cmd, NULL, '=');
|
|
if (eq && eq != cmd) {
|
|
eq[-1] = eq[1] = ' ';
|
|
}
|
|
r_str_replace_char (cmd, '\n', 0);
|
|
r_cons_printf ("%s\n", cmd);
|
|
free (cmd);
|
|
}
|
|
R_FREE (mflags);
|
|
}
|
|
r_list_foreach (c->fields, iter2, f) {
|
|
char *fn = r_str_newf ("field.%s.%s", c->name, f->name);
|
|
r_name_filter (fn, -1);
|
|
ut64 at = f->vaddr; // sym->vaddr + (f->vaddr & 0xffff);
|
|
r_cons_printf ("\"f %s = 0x%08"PFMT64x"\"\n", fn, at);
|
|
free (fn);
|
|
}
|
|
|
|
// C struct
|
|
r_cons_printf ("\"td struct %s {", c->name);
|
|
if (r_list_empty (c->fields)) {
|
|
// XXX workaround because we cant register empty structs yet
|
|
// XXX https://github.com/radareorg/radare2/issues/16342
|
|
r_cons_printf (" char empty[0];");
|
|
} else {
|
|
r_list_foreach (c->fields, iter2, f) {
|
|
char *n = objc_name_toc (f->name);
|
|
char *t = objc_type_toc (f->type);
|
|
r_cons_printf (" %s %s;", t, n);
|
|
free (t);
|
|
free (n);
|
|
}
|
|
}
|
|
r_cons_printf ("};\"\n");
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "classname", c->name);
|
|
pj_kN (pj, "addr", c->addr);
|
|
pj_ki (pj, "index", c->index);
|
|
if (c->super) {
|
|
pj_ks (pj, "visibility", r_str_get (c->visibility_str));
|
|
pj_ks (pj, "super", c->super);
|
|
}
|
|
pj_ka (pj, "methods");
|
|
r_list_foreach (c->methods, iter2, sym) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", sym->name);
|
|
if (sym->method_flags) {
|
|
char *mflags = r_core_bin_method_flags_str (sym->method_flags, mode);
|
|
pj_k (pj, "flags");
|
|
pj_j (pj, mflags);
|
|
free (mflags);
|
|
}
|
|
pj_kN (pj, "addr", sym->vaddr);
|
|
pj_end (pj);
|
|
}
|
|
pj_end (pj);
|
|
pj_ka (pj, "fields");
|
|
r_list_foreach (c->fields, iter3, f) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", f->name);
|
|
if (f->flags) {
|
|
char *mflags = r_core_bin_method_flags_str (f->flags, mode);
|
|
pj_k (pj, "flags");
|
|
pj_j (pj, mflags);
|
|
free (mflags);
|
|
}
|
|
pj_kN (pj, "addr", f->vaddr);
|
|
pj_end (pj);
|
|
}
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
} else {
|
|
int m = 0;
|
|
r_cons_printf ("0x%08"PFMT64x" [0x%08"PFMT64x" - 0x%08"PFMT64x"] %6"PFMT64d" class %d %s",
|
|
c->addr, at_min, at_max, (at_max - at_min), c->index, c->name);
|
|
if (c->super) {
|
|
r_cons_printf (" :: %s\n", c->super);
|
|
} else {
|
|
r_cons_newline ();
|
|
}
|
|
r_list_foreach (c->methods, iter2, sym) {
|
|
char *mflags = r_core_bin_method_flags_str (sym->method_flags, mode);
|
|
r_cons_printf ("0x%08"PFMT64x" method %d %s %s\n",
|
|
sym->vaddr, m, mflags, sym->dname? sym->dname: sym->name);
|
|
R_FREE (mflags);
|
|
m++;
|
|
}
|
|
}
|
|
free (name);
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static int bin_size(RCore *r, PJ *pj, int mode) {
|
|
ut64 size = r_bin_get_size (r->bin);
|
|
if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("%"PFMT64u"\n", size);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_n (pj, size);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("f bin_size @ %"PFMT64u"\n", size);
|
|
} else if (IS_MODE_SET (mode)) {
|
|
r_core_cmdf (r, "f bin_size @ %"PFMT64u"\n", size);
|
|
} else {
|
|
r_cons_printf ("%"PFMT64u"\n", size);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int bin_libs(RCore *r, PJ *pj, int mode) {
|
|
RListIter *iter;
|
|
char* lib;
|
|
int i = 0;
|
|
|
|
RList *libs = r_bin_get_libs (r->bin);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
} else {
|
|
if (!libs) {
|
|
return false;
|
|
}
|
|
if (IS_MODE_NORMAL (mode)) {
|
|
r_cons_println ("[Linked libraries]");
|
|
}
|
|
}
|
|
r_list_foreach (libs, iter, lib) {
|
|
if (IS_MODE_SET (mode)) {
|
|
// Nothing to set.
|
|
// TODO: load libraries with iomaps?
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("\"CCa entry0 %s\"\n", lib);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_s (pj, lib);
|
|
} else {
|
|
// simple and normal print mode
|
|
r_cons_println (lib);
|
|
}
|
|
i++;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
} else if (IS_MODE_NORMAL (mode)) {
|
|
const char *libstr = (i > 1)? "libraries": "library";
|
|
r_cons_printf ("\n%i %s\n", i, libstr);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void bin_mem_print(PJ *pj, RList *mems, int perms, int depth, int mode) {
|
|
RBinMem *mem;
|
|
RListIter *iter;
|
|
if (!mems) {
|
|
return;
|
|
}
|
|
r_list_foreach (mems, iter, mem) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", mem->name);
|
|
pj_ki (pj, "size", mem->size);
|
|
pj_kn (pj, "address", mem->addr);
|
|
pj_ks (pj, "flags", r_str_rwx_i (mem->perms & perms));
|
|
pj_end (pj);
|
|
} else if (IS_MODE_SIMPLE (mode)) {
|
|
r_cons_printf ("0x%08"PFMT64x"\n", mem->addr);
|
|
} else {
|
|
r_cons_printf ("0x%08"PFMT64x" +0x%04x %s %*s%-*s\n",
|
|
mem->addr, mem->size, r_str_rwx_i (mem->perms & perms),
|
|
depth, "", 20-depth, mem->name);
|
|
}
|
|
if (mem->mirrors) {
|
|
bin_mem_print (pj, mem->mirrors, mem->perms & perms, depth + 1, mode);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int bin_mem(RCore *r, PJ *pj, int mode) {
|
|
RList *mem = NULL;
|
|
if (!r) {
|
|
return false;
|
|
}
|
|
if (!IS_MODE_JSON (mode)) {
|
|
if (!(IS_MODE_RAD (mode) || IS_MODE_SET (mode))) {
|
|
r_cons_println ("[Memory]\n");
|
|
}
|
|
}
|
|
if (!(mem = r_bin_get_mem (r->bin))) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
pj_end (pj);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
bin_mem_print (pj, mem, 7, 0, R_MODE_JSON);
|
|
pj_end (pj);
|
|
return true;
|
|
} else if (!(IS_MODE_RAD (mode) || IS_MODE_SET (mode))) {
|
|
bin_mem_print (NULL, mem, 7, 0, mode);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void bin_pe_versioninfo(RCore *r, PJ *pj, int mode) {
|
|
Sdb *sdb = NULL;
|
|
int num_version = 0;
|
|
int num_stringtable = 0;
|
|
int num_string = 0;
|
|
const char *format_version = "bin/cur/info/vs_version_info/VS_VERSIONINFO%d";
|
|
const char *format_stringtable = "%s/string_file_info/stringtable%d";
|
|
const char *format_string = "%s/string%d";
|
|
if (!IS_MODE_JSON (mode)) {
|
|
r_cons_printf ("=== VS_VERSIONINFO ===\n\n");
|
|
} else {
|
|
pj_o (pj);
|
|
}
|
|
do {
|
|
char *path_version = r_str_newf (format_version, num_version);
|
|
sdb = sdb_ns_path (r->sdb, path_version, 0);
|
|
if (!sdb) {
|
|
free (path_version);
|
|
break;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ko (pj, "VS_FIXEDFILEINFO");
|
|
} else {
|
|
r_cons_printf ("# VS_FIXEDFILEINFO\n\n");
|
|
}
|
|
char *path_fixedfileinfo = r_str_newf ("%s/fixed_file_info", path_version);
|
|
sdb = sdb_ns_path (r->sdb, path_fixedfileinfo, 0);
|
|
free (path_fixedfileinfo);
|
|
if (!sdb) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
}
|
|
free (path_version);
|
|
break;
|
|
}
|
|
ut32 file_version_ms = sdb_num_get (sdb, "FileVersionMS", 0);
|
|
ut32 file_version_ls = sdb_num_get (sdb, "FileVersionLS", 0);
|
|
char *file_version = r_str_newf ("%u.%u.%u.%u", file_version_ms >> 16, file_version_ms & 0xFFFF,
|
|
file_version_ls >> 16, file_version_ls & 0xFFFF);
|
|
ut32 product_version_ms = sdb_num_get (sdb, "ProductVersionMS", 0);
|
|
ut32 product_version_ls = sdb_num_get (sdb, "ProductVersionLS", 0);
|
|
char *product_version = r_str_newf ("%u.%u.%u.%u", product_version_ms >> 16, product_version_ms & 0xFFFF,
|
|
product_version_ls >> 16, product_version_ls & 0xFFFF);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_kn (pj, "Signature", sdb_num_get (sdb, "Signature", 0));
|
|
pj_kn (pj, "StrucVersion", sdb_num_get (sdb, "StrucVersion", 0));
|
|
pj_ks (pj, "FileVersion", file_version);
|
|
pj_ks (pj, "ProductVersion", product_version);
|
|
pj_kn (pj, "FileFlagsMask", sdb_num_get (sdb, "FileFlagsMask", 0));
|
|
pj_kn (pj, "FileFlags", sdb_num_get (sdb, "FileFlags", 0));
|
|
pj_kn (pj, "FileOS", sdb_num_get (sdb, "FileOS", 0));
|
|
pj_kn (pj, "FileType", sdb_num_get (sdb, "FileType", 0));
|
|
pj_kn (pj, "FileSubType", sdb_num_get (sdb, "FileSubType", 0));
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_printf (" Signature: 0x%"PFMT64x"\n", sdb_num_get (sdb, "Signature", 0));
|
|
r_cons_printf (" StrucVersion: 0x%"PFMT64x"\n", sdb_num_get (sdb, "StrucVersion", 0));
|
|
r_cons_printf (" FileVersion: %s\n", file_version);
|
|
r_cons_printf (" ProductVersion: %s\n", product_version);
|
|
r_cons_printf (" FileFlagsMask: 0x%"PFMT64x"\n", sdb_num_get (sdb, "FileFlagsMask", 0));
|
|
r_cons_printf (" FileFlags: 0x%"PFMT64x"\n", sdb_num_get (sdb, "FileFlags", 0));
|
|
r_cons_printf (" FileOS: 0x%"PFMT64x"\n", sdb_num_get (sdb, "FileOS", 0));
|
|
r_cons_printf (" FileType: 0x%"PFMT64x"\n", sdb_num_get (sdb, "FileType", 0));
|
|
r_cons_printf (" FileSubType: 0x%"PFMT64x"\n", sdb_num_get (sdb, "FileSubType", 0));
|
|
r_cons_newline ();
|
|
}
|
|
free (file_version);
|
|
free (product_version);
|
|
#if 0
|
|
r_cons_printf (" FileDate: %d.%d.%d.%d\n",
|
|
sdb_num_get (sdb, "FileDateMS", 0) >> 16,
|
|
sdb_num_get (sdb, "FileDateMS", 0) & 0xFFFF,
|
|
sdb_num_get (sdb, "FileDateLS", 0) >> 16,
|
|
sdb_num_get (sdb, "FileDateLS", 0) & 0xFFFF);
|
|
#endif
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ko (pj, "StringTable");
|
|
} else {
|
|
r_cons_printf ("# StringTable\n\n");
|
|
}
|
|
for (num_stringtable = 0; sdb; num_stringtable++) {
|
|
char *path_stringtable = r_str_newf (format_stringtable, path_version, num_stringtable);
|
|
sdb = sdb_ns_path (r->sdb, path_stringtable, 0);
|
|
for (num_string = 0; sdb; num_string++) {
|
|
char *path_string = r_str_newf (format_string, path_stringtable, num_string);
|
|
sdb = sdb_ns_path (r->sdb, path_string, 0);
|
|
if (sdb) {
|
|
int lenkey = 0;
|
|
int lenval = 0;
|
|
ut8 *key_utf16 = sdb_decode (sdb_const_get (sdb, "key", 0), &lenkey);
|
|
ut8 *val_utf16 = sdb_decode (sdb_const_get (sdb, "value", 0), &lenval);
|
|
ut8 *key_utf8 = calloc (lenkey * 2, 1);
|
|
ut8 *val_utf8 = calloc (lenval * 2, 1);
|
|
if (r_str_utf16_to_utf8 (key_utf8, lenkey * 2, key_utf16, lenkey, true) < 0
|
|
|| r_str_utf16_to_utf8 (val_utf8, lenval * 2, val_utf16, lenval, true) < 0) {
|
|
eprintf ("Warning: Cannot decode utf16 to utf8\n");
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_ks (pj, (char*)key_utf8, (char*)val_utf8);
|
|
} else {
|
|
r_cons_printf (" %s: %s\n", (char*)key_utf8, (char*)val_utf8);
|
|
}
|
|
free (key_utf8);
|
|
free (val_utf8);
|
|
free (key_utf16);
|
|
free (val_utf16);
|
|
}
|
|
free (path_string);
|
|
}
|
|
free (path_stringtable);
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
}
|
|
num_version++;
|
|
free (path_version);
|
|
} while (sdb);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
|
|
static void bin_elf_versioninfo(RCore *r, PJ *pj, int mode) {
|
|
const char *format = "bin/cur/info/versioninfo/%s%d";
|
|
int num_versym;
|
|
int num_verneed = 0;
|
|
int num_version = 0;
|
|
Sdb *sdb = NULL;
|
|
const char *oValue = NULL;
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ka (pj, "versym");
|
|
}
|
|
for (num_versym = 0;; num_versym++) {
|
|
r_strf_var (versym_path, 128, format, "versym", num_versym);
|
|
if (!(sdb = sdb_ns_path (r->sdb, versym_path, 0))) {
|
|
break;
|
|
}
|
|
const ut64 addr = sdb_num_get (sdb, "addr", 0);
|
|
const ut64 offset = sdb_num_get (sdb, "offset", 0);
|
|
const ut64 link = sdb_num_get (sdb, "link", 0);
|
|
const ut64 num_entries = sdb_num_get (sdb, "num_entries", 0);
|
|
const char *const section_name = sdb_const_get (sdb, "section_name", 0);
|
|
const char *const link_section_name = sdb_const_get (sdb, "link_section_name", 0);
|
|
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "section_name", section_name);
|
|
pj_kn (pj, "address", addr);
|
|
pj_kn (pj, "offset", offset);
|
|
pj_kn (pj, "link", link);
|
|
pj_ks (pj, "link_section_name", link_section_name);
|
|
pj_ka (pj, "entries");
|
|
} else {
|
|
r_cons_printf ("Version symbols section '%s' contains %"PFMT64u" entries:\n", section_name, num_entries);
|
|
r_cons_printf (" Addr: 0x%08"PFMT64x" Offset: 0x%08"PFMT64x" Link: %x (%s)\n",
|
|
(ut64)addr, (ut64)offset, (ut32)link, link_section_name);
|
|
}
|
|
int i;
|
|
for (i = 0; i < num_entries; i++) {
|
|
r_strf_var (key, 32, "entry%d", i);
|
|
const char *const value = sdb_const_get (sdb, key, 0);
|
|
if (value) {
|
|
if (oValue && !strcmp (value, oValue)) {
|
|
continue;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_kn (pj, "idx", (ut64)i);
|
|
pj_ks (pj, "value", value);
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_printf (" 0x%08"PFMT64x": ", (ut64) i);
|
|
r_cons_printf ("%s\n", value);
|
|
}
|
|
oValue = value;
|
|
}
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_printf ("\n\n");
|
|
}
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
pj_ka (pj, "verneed");
|
|
}
|
|
|
|
do {
|
|
char *verneed_path = r_str_newf (format, "verneed", num_verneed++);
|
|
if (!(sdb = sdb_ns_path (r->sdb, verneed_path, 0))) {
|
|
break;
|
|
}
|
|
const char *const section_name = sdb_const_get (sdb, "section_name", 0);
|
|
const ut64 address = sdb_num_get (sdb, "addr", 0);
|
|
const ut64 offset = sdb_num_get (sdb, "offset", 0);
|
|
const ut64 link = sdb_num_get (sdb, "link", 0);
|
|
const char *const link_section_name = sdb_const_get (sdb, "link_section_name", 0);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "section_name", section_name);
|
|
pj_kn (pj, "address", address);
|
|
pj_kn (pj, "offset", offset);
|
|
pj_kn (pj, "link", link);
|
|
pj_ks (pj, "link_section_name", link_section_name);
|
|
pj_ka (pj, "entries");
|
|
} else {
|
|
r_cons_printf ("Version need section '%s' contains %d entries:\n",
|
|
section_name, (int)sdb_num_get (sdb, "num_entries", 0));
|
|
|
|
r_cons_printf (" Addr: 0x%08"PFMT64x, address);
|
|
|
|
r_cons_printf (" Offset: 0x%08"PFMT64x" Link to section: %"PFMT64d" (%s)\n",
|
|
offset, link, link_section_name);
|
|
}
|
|
for (num_version = 0;; num_version++) {
|
|
const char *filename = NULL;
|
|
int num_vernaux = 0;
|
|
|
|
r_strf_var (path_version, 64, "%s/version%d", verneed_path, num_version);
|
|
if (!(sdb = sdb_ns_path (r->sdb, path_version, 0))) {
|
|
break;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_kn (pj, "idx", sdb_num_get (sdb, "idx", 0));
|
|
pj_ki (pj, "vn_version", (int)sdb_num_get (sdb, "vn_version", 0));
|
|
} else {
|
|
r_cons_printf (" 0x%08"PFMT64x": Version: %d",
|
|
sdb_num_get (sdb, "idx", 0), (int)sdb_num_get (sdb, "vn_version", 0));
|
|
}
|
|
|
|
if ((filename = sdb_const_get (sdb, "file_name", 0))) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ks (pj, "file_name", filename);
|
|
} else {
|
|
r_cons_printf (" File: %s", filename);
|
|
}
|
|
}
|
|
const int cnt = (int)sdb_num_get (sdb, "cnt", 0);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ki (pj, "cnt", cnt);
|
|
} else {
|
|
r_cons_printf (" Cnt: %d\n", cnt);
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_ka (pj, "vernaux");
|
|
}
|
|
do {
|
|
char *path_vernaux = r_str_newf ("%s/vernaux%d", path_version, num_vernaux++);
|
|
sdb = sdb_ns_path (r->sdb, path_vernaux, 0);
|
|
free (path_vernaux);
|
|
if (!sdb) {
|
|
break;
|
|
}
|
|
const ut64 idx = sdb_num_get (sdb, "idx", 0);
|
|
const char *const name = sdb_const_get (sdb, "name", 0);
|
|
const char *const flags = sdb_const_get (sdb, "flags", 0);
|
|
const int version = (int)sdb_num_get (sdb, "version", 0);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_kn (pj, "idx", idx);
|
|
pj_ks (pj, "name", name);
|
|
pj_ks (pj, "flags", flags);
|
|
pj_ki (pj, "version", version);
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_printf (" 0x%08"PFMT64x": Name: %s", idx, name);
|
|
r_cons_printf (" Flags: %s Version: %d\n", flags, version);
|
|
}
|
|
} while (sdb);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
}
|
|
free (verneed_path);
|
|
} while (sdb);
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
|
|
static void bin_mach0_versioninfo(RCore *r) {
|
|
/* TODO */
|
|
}
|
|
|
|
static void bin_pe_resources(RCore *r, PJ *pj, int mode) {
|
|
Sdb *sdb = NULL;
|
|
int index = 0;
|
|
const char *pe_path = "bin/cur/info/pe_resource";
|
|
if (!(sdb = sdb_ns_path (r->sdb, pe_path, 0))) {
|
|
return;
|
|
}
|
|
if (IS_MODE_SET (mode)) {
|
|
r_flag_space_set (r->flags, R_FLAGS_FS_RESOURCES);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("fs resources\n");
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
}
|
|
while (true) {
|
|
r_strf_var (timestrKey, 32, "resource.%d.timestr", index);
|
|
r_strf_var (vaddrKey, 32, "resource.%d.vaddr", index);
|
|
r_strf_var (sizeKey, 32, "resource.%d.size", index);
|
|
r_strf_var (typeKey, 32, "resource.%d.type", index);
|
|
r_strf_var (languageKey, 32, "resource.%d.language", index);
|
|
r_strf_var (nameKey, 32, "resource.%d.name", index);
|
|
char *timestr = sdb_get (sdb, timestrKey, 0);
|
|
if (!timestr) {
|
|
break;
|
|
}
|
|
ut64 vaddr = sdb_num_get (sdb, vaddrKey, 0);
|
|
int size = (int)sdb_num_get (sdb, sizeKey, 0);
|
|
char *name = sdb_get (sdb, nameKey, 0);
|
|
char *type = sdb_get (sdb, typeKey, 0);
|
|
char *lang = sdb_get (sdb, languageKey, 0);
|
|
|
|
if (IS_MODE_SET (mode)) {
|
|
r_strf_var (name, 32, "resource.%d", index);
|
|
r_flag_set (r->flags, name, vaddr, size);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_printf ("f resource.%d %d 0x%08"PFMT64x"\n", index, size, vaddr);
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", name);
|
|
pj_ki (pj, "index", index);
|
|
pj_ks (pj, "type", type);
|
|
pj_kn (pj, "vaddr", vaddr);
|
|
pj_ki (pj, "size", size);
|
|
pj_ks (pj, "lang", lang);
|
|
pj_ks (pj, "timestamp", timestr);
|
|
pj_end (pj);
|
|
} else {
|
|
char humansz[8];
|
|
r_num_units (humansz, sizeof (humansz), size);
|
|
r_cons_printf ("Resource %d\n", index);
|
|
r_cons_printf (" name: %s\n", name);
|
|
r_cons_printf (" timestamp: %s\n", timestr);
|
|
r_cons_printf (" vaddr: 0x%08"PFMT64x"\n", vaddr);
|
|
r_cons_printf (" size: %s\n", humansz);
|
|
r_cons_printf (" type: %s\n", type);
|
|
r_cons_printf (" language: %s\n", lang);
|
|
}
|
|
|
|
R_FREE (timestr);
|
|
R_FREE (name);
|
|
R_FREE (type);
|
|
R_FREE (lang)
|
|
|
|
index++;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_end (pj);
|
|
} else if (IS_MODE_RAD (mode)) {
|
|
r_cons_println ("fs *");
|
|
}
|
|
}
|
|
|
|
static void bin_no_resources(RCore *r, PJ *pj, int mode) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_a (pj);
|
|
pj_end (pj);
|
|
}
|
|
}
|
|
|
|
static bool bin_resources(RCore *r, PJ *pj, int mode) {
|
|
const RBinInfo *info = r_bin_get_info (r->bin);
|
|
if (!info || !info->rclass) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
}
|
|
return false;
|
|
}
|
|
if (!strncmp ("pe", info->rclass, 2)) {
|
|
bin_pe_resources (r, pj, mode);
|
|
} else {
|
|
bin_no_resources (r, pj, mode);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool bin_versioninfo(RCore *r, PJ *pj, int mode) {
|
|
const RBinInfo *info = r_bin_get_info (r->bin);
|
|
if (!info || !info->rclass) {
|
|
if (IS_MODE_JSON (mode)) {
|
|
r_cons_print ("[]");
|
|
}
|
|
return false;
|
|
}
|
|
if (!strncmp ("pe", info->rclass, 2)) {
|
|
bin_pe_versioninfo (r, pj, mode);
|
|
} else if (!strncmp ("elf", info->rclass, 3)) {
|
|
bin_elf_versioninfo (r, pj, mode);
|
|
} else if (!strncmp ("mach0", info->rclass, 5)) {
|
|
bin_mach0_versioninfo (r); // TODO
|
|
if (IS_MODE_JSON (mode)) {
|
|
r_cons_print ("[]");
|
|
}
|
|
} else {
|
|
if (IS_MODE_JSON (mode)) {
|
|
r_cons_print ("[]");
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool bin_signature(RCore *r, PJ *pj, int mode) {
|
|
RBinFile *cur = r_bin_cur (r->bin);
|
|
RBinPlugin *plg = r_bin_file_cur_plugin (cur);
|
|
if (plg && plg->signature) {
|
|
const char *signature = plg->signature (cur, IS_MODE_JSON (mode));
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_k (pj, "signature");
|
|
pj_j (pj, signature);
|
|
pj_end (pj);
|
|
} else {
|
|
r_cons_println (signature);
|
|
}
|
|
free ((char*) signature);
|
|
return true;
|
|
}
|
|
if (IS_MODE_JSON (mode)) {
|
|
pj_o (pj);
|
|
pj_end (pj);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static bool bin_header(RCore *r, int mode) {
|
|
RBinFile *cur = r_bin_cur (r->bin);
|
|
RBinPlugin *plg = r_bin_file_cur_plugin (cur);
|
|
if (plg && plg->header) {
|
|
plg->header (cur);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_core_bin_info(RCore *core, int action, PJ *pj, int mode, int va, RCoreBinFilter *filter, const char *chksum) {
|
|
bool ret = true;
|
|
const char *name = NULL;
|
|
ut64 at = UT64_MAX, loadaddr = r_bin_get_laddr (core->bin);
|
|
if (filter && filter->offset) {
|
|
at = filter->offset;
|
|
}
|
|
if (filter && filter->name) {
|
|
name = filter->name;
|
|
}
|
|
|
|
// use our internal values for va
|
|
va = va ? VA_TRUE : VA_FALSE;
|
|
#if 0
|
|
if (r_config_get_i (core->config, "anal.strings")) {
|
|
r_core_cmd0 (core, "aar");
|
|
}
|
|
#endif
|
|
if ((action & R_CORE_BIN_ACC_RAW_STRINGS)) {
|
|
ret &= bin_raw_strings (core, pj, mode, va);
|
|
} else if ((action & R_CORE_BIN_ACC_STRINGS)) {
|
|
ret &= bin_strings (core, pj, mode, va);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_INFO)) {
|
|
ret &= bin_info (core, pj, mode, loadaddr);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_MAIN)) {
|
|
ret &= bin_main (core, pj, mode, va);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_DWARF)) {
|
|
ret &= bin_dwarf (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_PDB)) {
|
|
ret &= r_core_pdb_info (core, core->bin->file, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_SOURCE)) {
|
|
ret &= bin_source (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_ENTRIES)) {
|
|
ret &= bin_entry (core, pj, mode, loadaddr, va, false);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_INITFINI)) {
|
|
ret &= bin_entry (core, pj, mode, loadaddr, va, true);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_SECTIONS)) {
|
|
ret &= bin_sections (core, pj, mode, loadaddr, va, at, name, chksum, false);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_SEGMENTS)) {
|
|
ret &= bin_sections (core, pj, mode, loadaddr, va, at, name, chksum, true);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_SECTIONS_MAPPING)) {
|
|
ret &= bin_map_sections_to_segments (core->bin, pj, mode);
|
|
}
|
|
if (r_config_get_i (core->config, "bin.relocs")) {
|
|
if ((action & R_CORE_BIN_ACC_RELOCS)) {
|
|
ret &= bin_relocs (core, pj, mode, va);
|
|
}
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_LIBS)) {
|
|
ret &= bin_libs (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_IMPORTS)) { // 5s
|
|
ret &= bin_imports (core, pj, mode, va, name);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_EXPORTS)) {
|
|
ret &= bin_symbols (core, pj, mode, loadaddr, va, at, name, true, chksum);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_SYMBOLS)) { // 6s
|
|
ret &= bin_symbols (core, pj, mode, loadaddr, va, at, name, false, chksum);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_CLASSES)) { // 6s
|
|
ret &= bin_classes (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_TRYCATCH)) {
|
|
ret &= bin_trycatch (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_SIZE)) {
|
|
ret &= bin_size (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_MEM)) {
|
|
ret &= bin_mem (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_VERSIONINFO)) {
|
|
ret &= bin_versioninfo (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_RESOURCES)) {
|
|
ret &= bin_resources (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_SIGNATURE)) {
|
|
ret &= bin_signature (core, pj, mode);
|
|
}
|
|
if ((action & R_CORE_BIN_ACC_FIELDS)) {
|
|
if (IS_MODE_SIMPLE (mode)) {
|
|
if ((action & R_CORE_BIN_ACC_HEADER) || action & R_CORE_BIN_ACC_FIELDS) {
|
|
/* ignore mode, just for quiet/simple here */
|
|
ret &= bin_fields (core, NULL, 0, va);
|
|
}
|
|
} else {
|
|
if (IS_MODE_NORMAL (mode)) {
|
|
ret &= bin_header (core, mode);
|
|
} else {
|
|
if ((action & R_CORE_BIN_ACC_HEADER) || action & R_CORE_BIN_ACC_FIELDS) {
|
|
ret &= bin_fields (core, pj, mode, va);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
R_API bool r_core_bin_set_arch_bits(RCore *r, const char *name, const char *arch, ut16 bits) {
|
|
int fd = r_io_fd_get_current (r->io);
|
|
RIODesc *desc = r_io_desc_get (r->io, fd);
|
|
RBinFile *curfile, *binfile = NULL;
|
|
if (!name) {
|
|
if (!desc || !desc->name) {
|
|
return false;
|
|
}
|
|
name = desc->name;
|
|
}
|
|
/* Check if the arch name is a valid name */
|
|
if (!r_asm_is_valid (r->rasm, arch)) {
|
|
return false;
|
|
}
|
|
/* Find a file with the requested name/arch/bits */
|
|
binfile = r_bin_file_find_by_arch_bits (r->bin, arch, bits);
|
|
if (!binfile) {
|
|
return false;
|
|
}
|
|
if (!r_bin_use_arch (r->bin, arch, bits, name)) {
|
|
return false;
|
|
}
|
|
curfile = r_bin_cur (r->bin);
|
|
//set env if the binfile changed or we are dealing with xtr
|
|
if (curfile != binfile || binfile->curxtr) {
|
|
r_core_bin_set_cur (r, binfile);
|
|
return r_core_bin_set_env (r, binfile);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_core_bin_update_arch_bits(RCore *r) {
|
|
RBinFile *binfile = NULL;
|
|
const char *name = NULL, *arch = NULL;
|
|
ut16 bits = 0;
|
|
if (!r) {
|
|
return 0;
|
|
}
|
|
if (r->rasm) {
|
|
bits = r->rasm->bits;
|
|
if (r->rasm->cur) {
|
|
arch = r->rasm->cur->arch;
|
|
}
|
|
}
|
|
binfile = r_bin_cur (r->bin);
|
|
name = binfile ? binfile->file : NULL;
|
|
if (binfile && binfile->curxtr) {
|
|
r_anal_hint_clear (r->anal);
|
|
}
|
|
return r_core_bin_set_arch_bits (r, name, arch, bits);
|
|
}
|
|
|
|
R_API bool r_core_bin_raise(RCore *core, ut32 bfid) {
|
|
if (!r_bin_select_bfid (core->bin, bfid)) {
|
|
return false;
|
|
}
|
|
RBinFile *bf = r_bin_cur (core->bin);
|
|
if (bf) {
|
|
r_io_use_fd (core->io, bf->fd);
|
|
}
|
|
// it should be 0 to use r_io_use_fd in r_core_block_read
|
|
core->switch_file_view = 0;
|
|
return bf && r_core_bin_set_env (core, bf) && r_core_block_read (core);
|
|
}
|
|
|
|
R_API bool r_core_bin_delete(RCore *core, ut32 bf_id) {
|
|
if (bf_id == UT32_MAX) {
|
|
return false;
|
|
}
|
|
r_bin_file_delete (core->bin, bf_id);
|
|
RBinFile *bf = r_bin_file_at (core->bin, core->offset);
|
|
if (bf) {
|
|
r_io_use_fd (core->io, bf->fd);
|
|
}
|
|
core->switch_file_view = 0;
|
|
return bf && r_core_bin_set_env (core, bf) && r_core_block_read (core);
|
|
}
|
|
|
|
static bool r_core_bin_file_print(RCore *core, RBinFile *bf, PJ *pj, int mode) {
|
|
r_return_val_if_fail (core && bf && bf->o, false);
|
|
const char *name = bf ? bf->file : NULL;
|
|
(void)r_bin_get_info (core->bin); // XXX is this necssary for proper iniitialization
|
|
ut32 bin_sz = bf ? bf->size : 0;
|
|
// TODO: handle mode to print in json and r2 commands
|
|
|
|
switch (mode) {
|
|
case '*': {
|
|
char *n = __filterShell (name);
|
|
r_cons_printf ("oba 0x%08"PFMT64x" %s # %d\n", bf->o->boffset, n, bf->id);
|
|
free (n);
|
|
break;
|
|
}
|
|
case 'q':
|
|
r_cons_printf ("%d\n", bf->id);
|
|
break;
|
|
case 'j': {
|
|
pj_o (pj);
|
|
pj_ks (pj, "name", r_str_get (name));
|
|
pj_ki (pj, "iofd", bf->fd);
|
|
pj_ki (pj, "bfid", bf->id);
|
|
pj_ki (pj, "size", bin_sz);
|
|
pj_ko (pj, "obj");
|
|
RBinObject *obj = bf->o;
|
|
RBinInfo *info = obj->info;
|
|
ut8 bits = info ? info->bits : 0;
|
|
const char *asmarch = r_config_get (core->config, "asm.arch");
|
|
const char *arch = info
|
|
? info->arch
|
|
? info->arch
|
|
: asmarch
|
|
: "unknown";
|
|
pj_ks (pj, "arch", arch);
|
|
pj_ki (pj, "bits", bits);
|
|
pj_kN (pj, "binoffset", obj->boffset);
|
|
pj_kN (pj, "objsize", obj->obj_size);
|
|
pj_end (pj);
|
|
pj_end (pj);
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
RBinInfo *info = bf->o->info;
|
|
ut8 bits = info ? info->bits : 0;
|
|
const char *asmarch = r_config_get (core->config, "asm.arch");
|
|
const char *arch = info ? info->arch ? info->arch: asmarch: "unknown";
|
|
const char *curstr = (core->allbins || bf == r_bin_cur (core->bin)) ? "*": "-";
|
|
r_cons_printf ("%s %d %d %s-%d ba:0x%08"PFMT64x" sz:%"PFMT64d" %s\n",
|
|
curstr, bf->id, bf->fd, arch, bits, bf->o->baddr, bf->o->size, name);
|
|
}
|
|
break;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_core_bin_list(RCore *core, int mode) {
|
|
// list all binfiles and there objects and there archs
|
|
RListIter *iter;
|
|
RBinFile *binfile = NULL; //, *cur_bf = r_bin_cur (core->bin) ;
|
|
RBin *bin = core->bin;
|
|
const RList *binfiles = bin ? bin->binfiles: NULL;
|
|
if (!binfiles) {
|
|
return false;
|
|
}
|
|
PJ *pj = NULL;
|
|
if (mode == 'j') {
|
|
pj = r_core_pj_new (core);
|
|
if (!pj) {
|
|
return 0;
|
|
}
|
|
pj_a (pj);
|
|
}
|
|
r_list_foreach (binfiles, iter, binfile) {
|
|
r_core_bin_file_print (core, binfile, pj, mode);
|
|
}
|
|
if (mode == 'j') {
|
|
pj_end (pj);
|
|
r_cons_print (pj_string (pj));
|
|
pj_free (pj);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API char *r_core_bin_method_flags_str(ut64 flags, int mode) {
|
|
int i;
|
|
|
|
RStrBuf *buf = r_strbuf_new ("");
|
|
if (IS_MODE_SET (mode) || IS_MODE_RAD (mode)) {
|
|
if (!flags) {
|
|
goto out;
|
|
}
|
|
|
|
for (i = 0; i < 64; i++) {
|
|
ut64 flag = flags & (1ULL << i);
|
|
if (flag) {
|
|
const char *flag_string = r_bin_get_meth_flag_string (flag, false);
|
|
if (flag_string) {
|
|
r_strbuf_appendf (buf, ".%s", flag_string);
|
|
}
|
|
}
|
|
}
|
|
} else if (IS_MODE_JSON (mode)) {
|
|
if (!flags) {
|
|
r_strbuf_append (buf, "[]");
|
|
goto out;
|
|
}
|
|
PJ *pj = pj_new ();
|
|
pj_a (pj);
|
|
for (i = 0; i < 64; i++) {
|
|
ut64 flag = flags & (1ULL << i);
|
|
if (flag) {
|
|
const char *flag_string = r_bin_get_meth_flag_string (flag, false);
|
|
if (flag_string) {
|
|
pj_s (pj, flag_string);
|
|
} else {
|
|
r_strf_var (numstr, 32, "0x%08"PFMT64x, flag);
|
|
pj_s (pj, numstr);
|
|
}
|
|
}
|
|
}
|
|
pj_end (pj);
|
|
r_strbuf_append (buf, pj_string (pj));
|
|
pj_free (pj);
|
|
} else {
|
|
int pad_len = 4; //TODO: move to a config variable
|
|
int len = 0;
|
|
if (!flags) {
|
|
goto padding;
|
|
}
|
|
for (i = 0; i < 64; i++) {
|
|
ut64 flag = flags & (1ULL << i);
|
|
if (flag) {
|
|
const char *flag_string = r_bin_get_meth_flag_string (flag, true);
|
|
if (flag_string) {
|
|
r_strbuf_append (buf, flag_string);
|
|
} else {
|
|
r_strbuf_append (buf, "?");
|
|
}
|
|
len++;
|
|
}
|
|
}
|
|
padding:
|
|
for ( ; len < pad_len; len++) {
|
|
r_strbuf_append (buf, " ");
|
|
}
|
|
}
|
|
out:
|
|
return r_strbuf_drain (buf);
|
|
}
|
|
|
|
R_API bool r_core_bin_rebase(RCore *core, ut64 baddr) {
|
|
r_return_val_if_fail (core && core->bin, false);
|
|
if (!core->bin->cur || baddr == UT64_MAX) {
|
|
return false;
|
|
}
|
|
RBinFile *bf = core->bin->cur;
|
|
bf->o->baddr = baddr;
|
|
bf->o->loadaddr = baddr;
|
|
r_bin_object_set_items (bf, bf->o);
|
|
return true;
|
|
}
|
|
|