radare2/libr/anal/meta.c

630 lines
16 KiB
C

/* radare - LGPL - Copyright 2008-2022 - nibble, pancake, thestr4ng3r */
#include <r_anal.h>
#include <r_core.h>
static bool item_matches_filter(RAnalMetaItem *item, RAnalMetaType type, R_NULLABLE const RSpace *space) {
return (type == R_META_TYPE_ANY || item->type == type) && (!space || item->space == space);
}
typedef struct {
RAnalMetaType type;
const RSpace *space;
RIntervalNode *node;
} FindCtx;
static bool find_node_cb(RIntervalNode *node, void *user) {
FindCtx *ctx = user;
if (item_matches_filter (node->data, ctx->type, ctx->space)) {
ctx->node = node;
return false;
}
return true;
}
static RIntervalNode *find_node_at(RAnal *anal, RAnalMetaType type, R_NULLABLE const RSpace *space, ut64 addr) {
FindCtx ctx = {
.type = type,
.space = space,
.node = NULL
};
r_interval_tree_all_at (&anal->meta, addr, find_node_cb, &ctx);
return ctx.node;
}
static RIntervalNode *find_node_in(RAnal *anal, RAnalMetaType type, R_NULLABLE const RSpace *space, ut64 addr) {
FindCtx ctx = {
.type = type,
.space = space,
.node = NULL
};
r_interval_tree_all_in (&anal->meta, addr, true, find_node_cb, &ctx);
return ctx.node;
}
typedef struct {
RAnalMetaType type;
const RSpace *space;
RPVector/*RIntervalNode*/ *result;
} CollectCtx;
static bool collect_nodes_cb(RIntervalNode *node, void *user) {
CollectCtx *ctx = user;
if (item_matches_filter (node->data, ctx->type, ctx->space)) {
r_pvector_push (ctx->result, node);
}
return true;
}
static RPVector *collect_nodes_at(RAnal *anal, RAnalMetaType type, R_NULLABLE const RSpace *space, ut64 addr) {
CollectCtx ctx = {
.type = type,
.space = space,
.result = r_pvector_new (NULL)
};
if (!ctx.result) {
return NULL;
}
r_interval_tree_all_at (&anal->meta, addr, collect_nodes_cb, &ctx);
return ctx.result;
}
static RPVector *collect_nodes_in(RAnal *anal, RAnalMetaType type, R_NULLABLE const RSpace *space, ut64 addr) {
CollectCtx ctx = {
.type = type,
.space = space,
.result = r_pvector_new (NULL)
};
if (!ctx.result) {
return NULL;
}
r_interval_tree_all_in (&anal->meta, addr, true, collect_nodes_cb, &ctx);
return ctx.result;
}
static RPVector *collect_nodes_intersect(RAnal *anal, RAnalMetaType type, R_NULLABLE const RSpace *space, ut64 start, ut64 end) {
CollectCtx ctx = {
.type = type,
.space = space,
.result = r_pvector_new (NULL)
};
if (!ctx.result) {
return NULL;
}
r_interval_tree_all_intersect (&anal->meta, start, end, true, collect_nodes_cb, &ctx);
return ctx.result;
}
static bool meta_set(RAnal *a, RAnalMetaType type, int subtype, ut64 from, ut64 to, const char *str) {
if (to < from) {
return false;
}
RSpace *space = r_spaces_current (&a->meta_spaces);
RIntervalNode *node = find_node_at (a, type, space, from);
RAnalMetaItem *item = node ? node->data : R_NEW0 (RAnalMetaItem);
if (!item) {
return false;
}
item->type = type;
item->subtype = subtype;
item->space = space;
free (item->str);
item->str = str ? strdup (str) : NULL;
if (str && !item->str) {
if (!node) { // If we just created this
free (item);
}
return false;
}
R_DIRTY (a);
if (!node) {
r_interval_tree_insert (&a->meta, from, to, item);
} else if (node->end != to) {
r_interval_tree_resize (&a->meta, node, from, to);
}
return true;
}
R_API bool r_meta_set_string(RAnal *a, RAnalMetaType type, ut64 addr, const char *s) {
return meta_set (a, type, 0, addr, addr, s);
}
R_API const char *r_meta_get_string(RAnal *a, RAnalMetaType type, ut64 addr) {
RIntervalNode *node = find_node_at (a, type, r_spaces_current (&a->meta_spaces), addr);
if (!node) {
return NULL;
}
RAnalMetaItem *item = node->data;
return item->str;
}
static void del(RAnal *a, RAnalMetaType type, const RSpace *space, ut64 addr, ut64 size) {
RPVector *victims = NULL;
if (size == UT64_MAX) {
// delete everything
victims = r_pvector_new (NULL);
if (!victims) {
return;
}
RIntervalTreeIter it;
RAnalMetaItem *item;
r_interval_tree_foreach (&a->meta, it, item) {
if (item_matches_filter (item, type, space)) {
r_pvector_push (victims, r_interval_tree_iter_get (&it));
}
}
} else {
ut64 end = size ? addr + size - 1 : addr;
if (end < addr) {
end = UT64_MAX;
}
victims = collect_nodes_intersect (a, type, space, addr, end);
if (!victims) {
return;
}
}
void **it;
r_pvector_foreach (victims, it) {
r_interval_tree_delete (&a->meta, *it, true);
}
r_pvector_free (victims);
}
R_API void r_meta_del(RAnal *a, RAnalMetaType type, ut64 addr, ut64 size) {
del (a, type, r_spaces_current (&a->meta_spaces), addr, size);
}
R_API bool r_meta_set(RAnal *a, RAnalMetaType type, ut64 addr, ut64 size, const char *str) {
return r_meta_set_with_subtype (a, type, 0, addr, size, str);
}
R_API bool r_meta_set_with_subtype(RAnal *m, RAnalMetaType type, int subtype, ut64 addr, ut64 size, const char *str) {
r_return_val_if_fail (m && size, false);
ut64 end = addr + size - 1;
if (end < addr) {
end = UT64_MAX;
}
return meta_set (m, type, subtype, addr, end, str);
}
R_API RAnalMetaItem *r_meta_get_at(RAnal *a, ut64 addr, RAnalMetaType type, R_OUT R_NULLABLE ut64 *size) {
RIntervalNode *node = find_node_at (a, type, r_spaces_current (&a->meta_spaces), addr);
if (node && size) {
*size = r_meta_item_size (node->start, node->end);
}
return node ? node->data : NULL;
}
R_API RIntervalNode *r_meta_get_in(RAnal *a, ut64 addr, RAnalMetaType type) {
return find_node_in (a, type, r_spaces_current (&a->meta_spaces), addr);
}
R_API RPVector/*<RIntervalNode<RMetaItem> *>*/ *r_meta_get_all_at(RAnal *a, ut64 at) {
return collect_nodes_at (a, R_META_TYPE_ANY, r_spaces_current (&a->meta_spaces), at);
}
R_API RPVector *r_meta_get_all_in(RAnal *a, ut64 at, RAnalMetaType type) {
return collect_nodes_in (a, type, r_spaces_current (&a->meta_spaces), at);
}
R_API RPVector *r_meta_get_all_intersect(RAnal *a, ut64 start, ut64 size, RAnalMetaType type) {
r_return_val_if_fail (size, NULL);
ut64 end = start + size - 1;
if (end < start) {
end = UT64_MAX;
}
return collect_nodes_intersect (a, type, r_spaces_current (&a->meta_spaces), start, end);
}
R_API const char *r_meta_type_tostring(int type) {
// XXX: use type as '%c'
switch (type) {
case R_META_TYPE_DATA: return "Cd";
case R_META_TYPE_CODE: return "Cc";
case R_META_TYPE_STRING: return "Cs";
case R_META_TYPE_FORMAT: return "Cf";
case R_META_TYPE_MAGIC: return "Cm";
case R_META_TYPE_HIDE: return "Ch";
case R_META_TYPE_COMMENT: return "CCu";
case R_META_TYPE_RUN: return "Cr"; // not in C? help
case R_META_TYPE_HIGHLIGHT: return "ecHi"; // not in C?
case R_META_TYPE_VARTYPE: return "Ct";
}
return "# unknown meta # ";
}
R_API void r_meta_print(RAnal *a, RAnalMetaItem *d, ut64 start, ut64 size, int rad, PJ *pj, bool show_full) {
r_return_if_fail (!(rad == 'j' && !pj)); // rad == 'j' => pj
char *pstr, *base64_str;
RCore *core = a->coreb.core;
bool esc_bslash = core ? core->print->esc_bslash : false;
if (r_spaces_current (&a->meta_spaces) &&
r_spaces_current (&a->meta_spaces) != d->space) {
return;
}
char *str = NULL;
if (d->str) {
if (d->type == R_META_TYPE_STRING) {
if (d->subtype == R_STRING_ENC_UTF8) {
str = r_str_escape_utf8 (d->str, false, esc_bslash);
} else {
if (!d->subtype) { /* temporary legacy workaround */
esc_bslash = false;
}
str = r_str_escape_latin1 (d->str, false, esc_bslash, false);
}
} else {
str = r_str_escape (d->str);
}
}
if (str || d->type == R_META_TYPE_DATA) {
if (d->type == R_META_TYPE_STRING && !*str) {
free (str);
return;
}
if (!str) {
pstr = "";
} else if (d->type == 'f') {
pstr = str;
} else if (d->type == 's') {
pstr = str;
} else if (d->type == 't') {
// Sanitize (don't escape) Ct comments so we can see "char *", etc.
free (str);
str = strdup (d->str);
r_str_sanitize (str);
pstr = str;
} else if (d->type != 'C') {
r_name_filter (str, 0);
pstr = str;
} else {
pstr = d->str;
}
// r_str_sanitize (str);
switch (rad) {
case 'j':
pj_o (pj);
pj_kn (pj, "offset", start);
pj_ks (pj, "type", r_meta_type_tostring (d->type));
if (d->type == 'H') {
pj_k (pj, "color");
ut8 r = 0, g = 0, b = 0, A = 0;
const char *esc = strchr (d->str, '\x1b');
if (esc) {
r_cons_rgb_parse (esc, &r, &g, &b, &A);
char *rgb_str = r_cons_rgb_tostring (r, g, b);
base64_str = r_base64_encode_dyn (rgb_str, -1);
if (d->type == 's' && base64_str) {
pj_s (pj, base64_str);
free (base64_str);
} else {
pj_s (pj, rgb_str);
}
free (rgb_str);
} else {
pj_s (pj, str);
}
} else {
pj_k (pj, "name");
if (d->type == 's' && (base64_str = r_base64_encode_dyn (d->str, -1))) {
pj_s (pj, base64_str);
} else {
pj_s (pj, str);
}
}
if (d->type == 'd') {
pj_kn (pj, "size", size);
} else if (d->type == 's') {
const char *enc;
switch (d->subtype) {
case R_STRING_ENC_UTF8:
enc = "utf8";
break;
case 0: /* temporary legacy encoding */
enc = "iz";
break;
default:
enc = "latin1";
}
pj_ks (pj, "enc", enc);
pj_kb (pj, "ascii", r_str_is_ascii (d->str));
}
pj_end (pj);
break;
case 0:
case 1:
case '*':
default:
switch (d->type) {
case R_META_TYPE_COMMENT:
{
const char *type = r_meta_type_tostring (d->type);
char *s = sdb_encode ((const ut8*)pstr, -1);
if (!s) {
s = strdup (pstr);
}
if (rad) {
if (!strcmp (type, "CCu")) {
a->cb_printf ("%s base64:%s @ 0x%08" PFMT64x "\n",
type, s, start);
} else {
a->cb_printf ("%s %s @ 0x%08" PFMT64x "\n",
type, pstr, start);
}
} else {
if (!strcmp (type, "CCu")) {
char *mys = r_str_escape (pstr);
a->cb_printf ("0x%08" PFMT64x " %s \"%s\"\n",
start, type, mys);
free (mys);
} else {
a->cb_printf ("0x%08"PFMT64x" %s \"%s\"\n",
start, type, pstr);
}
}
free (s);
}
break;
case R_META_TYPE_STRING:
if (rad) {
char cmd[] = "Cs#";
switch (d->subtype) {
case 'a':
case '8':
cmd[2] = d->subtype;
break;
default:
cmd[2] = 0;
}
a->cb_printf ("%s %" PFMT64u " @ 0x%08" PFMT64x " # %s\n",
cmd, size, start, pstr);
} else {
const char *enc;
switch (d->subtype) {
case '8':
enc = "utf8";
break;
default:
enc = r_str_is_ascii (d->str)? "ascii": "latin1";
}
if (show_full) {
a->cb_printf ("0x%08" PFMT64x " %s[%" PFMT64u "] \"%s\"\n",
start, enc, size, pstr);
} else {
a->cb_printf ("%s[%" PFMT64u "] \"%s\"\n",
enc, size, pstr);
}
}
break;
case R_META_TYPE_HIDE:
case R_META_TYPE_DATA:
if (rad) {
a->cb_printf ("%s %" PFMT64u " @ 0x%08" PFMT64x "\n",
r_meta_type_tostring (d->type),
size, start);
} else {
if (show_full) {
const char *dtype = d->type == 'h'? "hidden": "data";
a->cb_printf ("0x%08" PFMT64x " %s %s %"PFMT64u"\n",
start, dtype,
r_meta_type_tostring (d->type),
size);
} else {
a->cb_printf ("%" PFMT64u "\n", size);
}
}
break;
case R_META_TYPE_MAGIC:
case R_META_TYPE_FORMAT:
if (rad) {
a->cb_printf ("%s %" PFMT64u " %s @ 0x%08" PFMT64x "\n",
r_meta_type_tostring (d->type),
size, pstr, start);
} else {
if (show_full) {
const char *dtype = d->type == 'm'? "magic": "format";
a->cb_printf ("0x%08" PFMT64x " %s %" PFMT64u " %s\n",
start, dtype, size, pstr);
} else {
a->cb_printf ("%" PFMT64u " %s\n", size, pstr);
}
}
break;
case R_META_TYPE_VARTYPE:
if (rad) {
a->cb_printf ("%s %s @ 0x%08" PFMT64x "\n",
r_meta_type_tostring (d->type), pstr, start);
} else {
a->cb_printf ("0x%08" PFMT64x " %s\n", start, pstr);
}
break;
case R_META_TYPE_HIGHLIGHT:
{
ut8 r = 0, g = 0, b = 0, A = 0;
const char *esc = strchr (d->str, '\x1b');
r_cons_rgb_parse (esc, &r, &g, &b, &A);
a->cb_printf ("%s rgb:%02x%02x%02x @ 0x%08" PFMT64x "\n",
r_meta_type_tostring (d->type), r, g, b, start);
// TODO: d->size
}
break;
default:
if (rad) {
a->cb_printf ("%s %" PFMT64u " 0x%08" PFMT64x " # %s\n",
r_meta_type_tostring (d->type),
size, start, pstr);
} else {
// TODO: use b64 here
a->cb_printf ("0x%08" PFMT64x " array[%" PFMT64u "] %s %s\n",
start, size,
r_meta_type_tostring (d->type), pstr);
}
break;
}
break;
}
if (str) {
free (str);
}
}
}
R_API void r_meta_print_list_at(RAnal *a, ut64 addr, int rad, const char *tq) {
RPVector *nodes = collect_nodes_at (a, R_META_TYPE_ANY, r_spaces_current (&a->meta_spaces), addr);
if (!nodes) {
return;
}
void **it;
r_pvector_foreach (nodes, it) {
RIntervalNode *node = *it;
r_meta_print (a, node->data, node->start, r_meta_node_size (node), rad, NULL, true);
}
r_pvector_free (nodes);
}
static void print_meta_list(RAnal *a, int type, int rad, ut64 addr, const char *tq) {
PJ *pj = NULL;
RTable *t = NULL;
if (rad == ',') {
t = r_table_new ("meta");
RTableColumnType *s = r_table_type ("string");
RTableColumnType *n = r_table_type ("number");
r_table_add_column (t, n, "addr", 0);
r_table_add_column (t, n, "size", 0);
r_table_add_column (t, s, "type", 0);
r_table_add_column (t, s, "string", 0);
} else if (rad == 'j') {
pj = pj_new ();
if (!pj) {
return;
}
pj_a (pj);
}
RAnalFunction *fcn = NULL;
if (addr != UT64_MAX) {
fcn = r_anal_get_fcn_in (a, addr, 0);
if (!fcn) {
goto beach;
}
}
RIntervalTreeIter it;
RAnalMetaItem *item;
r_interval_tree_foreach (&a->meta, it, item) {
RIntervalNode *node = r_interval_tree_iter_get (&it);
if (type != R_META_TYPE_ANY && item->type != type) {
continue;
}
if (fcn && !r_anal_function_contains (fcn, node->start)) {
continue;
}
if (t) {
const char *type = r_meta_type_tostring (item->type);
const char *name = item->str;
r_table_add_rowf (t, "xxss",
node->start,
r_meta_node_size (node),
type, name);
} else {
r_meta_print (a, item, node->start, r_meta_node_size (node), rad, pj, true);
}
}
beach:
if (t && tq) {
r_table_query (t, tq);
}
if (!tq || !strstr (tq, "?")) {
if (t) {
char *s = r_table_tostring (t);
r_cons_printf ("%s\n", s);
free (s);
} else if (pj) {
pj_end (pj);
r_cons_printf ("%s\n", pj_string (pj));
pj_free (pj);
pj = NULL;
}
}
pj_free (pj);
}
R_API void r_meta_print_list_all(RAnal *a, int type, int rad, const char *tq) {
print_meta_list (a, type, rad, UT64_MAX, tq);
}
R_API void r_meta_print_list_in_function(RAnal *a, int type, int rad, ut64 addr, const char *tq) {
print_meta_list (a, type, rad, addr, tq);
}
R_API void r_meta_rebase(RAnal *anal, ut64 diff) {
if (!diff) {
return;
}
RIntervalTree old = anal->meta;
r_interval_tree_init (&anal->meta, old.free);
RIntervalTreeIter it;
RAnalMetaItem *item;
r_interval_tree_foreach (&old, it, item) {
RIntervalNode *node = r_interval_tree_iter_get (&it);
ut64 newstart = node->start + diff;
ut64 newend = node->end + diff;
if (newend < newstart) {
// Can't rebase this
newstart = node->start;
newend = node->end;
}
r_interval_tree_insert (&anal->meta, newstart, newend, item);
}
old.free = NULL;
r_interval_tree_fini (&old);
}
R_API void r_meta_space_unset_for(RAnal *a, const RSpace *space) {
del (a, R_META_TYPE_ANY, space, 0, UT64_MAX);
}
R_API ut64 r_meta_get_size(RAnal *a, RAnalMetaType type) {
r_return_val_if_fail (a, 0);
if (!a->meta.root) {
return 0;
}
ut64 sum = 0;
RIntervalTreeIter it;
RAnalMetaItem *item;
RIntervalNode *prev = NULL;
r_interval_tree_foreach (&a->meta, it, item) {
RIntervalNode *node = r_interval_tree_iter_get (&it);
if (type != R_META_TYPE_ANY && item->type != type) {
continue;
}
ut64 start = R_MAX (prev ? prev->end : 0, node->start);
sum += node->end - start + 1;
prev = node;
}
return sum;
}
R_API int r_meta_space_count_for(RAnal *a, const RSpace *space) {
int r = 0;
RIntervalTreeIter it;
RAnalMetaItem *item;
r_interval_tree_foreach (&a->meta, it, item) {
if (item->space == space) {
r++;
}
}
return r;
}
R_API void r_meta_set_data_at(RAnal *a, ut64 addr, ut64 wordsz) {
r_return_if_fail (wordsz);
r_meta_set (a, R_META_TYPE_DATA, addr, wordsz, NULL);
}