radare2/libr/anal/meta.c

318 lines
8.1 KiB
C
Raw Normal View History

2014-01-15 00:56:28 +00:00
/* radare - LGPL - Copyright 2008-2014 - nibble, pancake */
2014-03-04 11:23:52 +00:00
// TODO: rename to r_anal_meta_get() ??
#if 0
TODO
====
- handle sync to synchronize all the data on disk.
- actually listing only works in memory
- array_add doesnt needs index, right?
2014-05-08 21:20:42 +00:00
- remove unused arguments from r_meta_find (where ?)
- implement r_meta_find
#endif
#include <r_anal.h>
#include <r_print.h>
2014-03-04 11:23:52 +00:00
#define META_RANGE_BASE(x) ((x)>>12)
#define META_RANGE_SIZE 0xfff
#undef DB
#define DB a->sdb_meta
static char *meta_inrange_get (RAnal *a, ut64 addr, int size) {
2014-03-04 11:23:52 +00:00
char key[64];
ut64 base, base2;
base = META_RANGE_BASE (addr);
base2 = META_RANGE_BASE (addr+size);
// return string array of all the offsets where there are stuff
for (; base<base2; base += META_RANGE_SIZE) {
snprintf (key, sizeof (key)-1, "range.0x%"PFMT64x, base);
sdb_array_get (DB, key, 0, 0);
}
return NULL;
}
static int meta_inrange_add (RAnal *a, ut64 addr, int size) {
int set = 0;
char key[64];
ut64 base, base2;
base = META_RANGE_BASE (addr);
base2 = META_RANGE_BASE (addr+size);
for (; base<base2; base += META_RANGE_SIZE) {
snprintf (key, sizeof (key)-1, "range.0x%"PFMT64x, base);
2014-03-26 03:16:03 +00:00
if (sdb_array_add_num (DB, key, addr, 0))
2014-03-04 11:23:52 +00:00
set = 1;
}
return set;
}
static int meta_inrange_del (RAnal *a, ut64 addr, int size) {
int set = 0;
char key[64];
ut64 base, base2;
base = META_RANGE_BASE (addr);
base2 = META_RANGE_BASE (addr+size);
// TODO: optimize this thing?
for (; base<base2; base += META_RANGE_SIZE) {
snprintf (key, sizeof (key)-1, "range.0x%"PFMT64x, base);
if (sdb_array_remove_num (DB, key, addr, 0))
2014-03-04 11:23:52 +00:00
set = 1;
}
//sdb_array_del (DB);
return set;
}
2014-03-04 11:23:52 +00:00
// TODO: Add APIs to resize meta? nope, just del and add
R_API int r_meta_set_string(RAnal *a, int type, ut64 addr, const char *s) {
2014-03-04 11:23:52 +00:00
char key[100], val[2048], *e_str;
int ret;
ut64 size;
snprintf (key, sizeof (key)-1, "meta.%c", type);
2014-03-26 03:16:03 +00:00
sdb_array_add_num (DB, key, addr, 0);
2014-03-04 11:23:52 +00:00
snprintf (key, sizeof (key)-1, "meta.%c.0x%"PFMT64x, type, addr);
size = sdb_array_get_num (DB, key, 0, 0);
if (!size) {
size = strlen (s);
meta_inrange_add (a, addr, size);
2014-03-04 11:23:52 +00:00
ret = R_TRUE;
} else ret = R_FALSE;
e_str = sdb_encode ((const void*)s, -1);
2014-03-04 11:23:52 +00:00
snprintf (val, sizeof (val)-1, "%d,%s", (int)size, e_str);
sdb_set (DB, key, val, 0);
free ((void*)e_str);
return ret;
}
R_API char *r_meta_get_string(RAnal *a, int type, ut64 addr) {
2014-03-04 11:23:52 +00:00
char key[100];
const char *k, *p;
snprintf (key, sizeof (key)-1, "meta.%c.0x%"PFMT64x, 'C', addr);
k = sdb_const_get (DB, key, NULL);
if (!k) return NULL;
p = strchr (k, SDB_RS);
if (!p) return NULL;
k = p+1;
// TODO : comment append has been deprecated
return (char *)sdb_decode (k, NULL);
}
2014-03-04 11:23:52 +00:00
R_API int r_meta_del(RAnal *a, int type, ut64 addr, ut64 size, const char *str) {
char key[100], key2[100], *dtr, *s, *p, *next;
2014-03-04 11:23:52 +00:00
const char *ptr;
int i;
2014-03-04 11:23:52 +00:00
if (size == UT64_MAX) {
// FULL CLEANUP
// XXX: this thing ignores the type
if (type == R_META_TYPE_ANY) {
sdb_reset (DB);
} else {
snprintf (key, sizeof (key)-1, "meta.%c", type);
dtr = sdb_get (DB, key, 0);
for (p = dtr; p; p = next) {
2014-03-26 03:16:03 +00:00
s = sdb_anext (p, &next);
snprintf (key, sizeof (key)-1,
"meta.%c.0x%"PFMT64x,
2014-03-04 11:23:52 +00:00
type, sdb_atoi (s));
eprintf ("--> %s\n", key);
sdb_unset (DB, key, 0);
if (!next) break;
2014-03-04 11:23:52 +00:00
}
free (dtr);
}
return R_FALSE;
}
meta_inrange_del (a, addr, size);
snprintf (key, sizeof (key)-1, type==R_META_TYPE_COMMENT ? "meta.C.0x%"PFMT64x : "meta.0x%"PFMT64x, addr);
2014-03-04 11:23:52 +00:00
ptr = sdb_const_get (DB, key, 0);
if (ptr) {
for (i=0; ptr[i]; i++) {
if (ptr[i] != SDB_RS) {
snprintf (key2, sizeof (key2)-1,
"meta.%c.0x%"PFMT64x, ptr[i], addr);
sdb_unset (DB, key2, 0);
}
2014-03-04 11:23:52 +00:00
}
}
sdb_unset (DB, key, 0);
return R_FALSE;
}
2014-01-23 02:28:25 +00:00
R_API int r_meta_cleanup(RAnal *a, ut64 from, ut64 to) {
return r_meta_del (a, R_META_TYPE_ANY, from, (to-from), NULL);
}
R_API void r_meta_item_free(void *_item) {
2014-01-23 03:34:59 +00:00
RAnalMetaItem *item = _item;
free (item);
}
2014-01-23 03:34:59 +00:00
R_API RAnalMetaItem *r_meta_item_new(int type) {
RAnalMetaItem *mi = R_NEW (RAnalMetaItem);
memset (mi, 0, sizeof (RAnalMetaItem));
mi->type = type;
return mi;
}
R_API int r_meta_add(RAnal *a, int type, ut64 from, ut64 to, const char *str) {
int exists;
2014-03-04 11:23:52 +00:00
char *e_str, key[100], val[2048];
if (from>to)
2014-03-04 11:23:52 +00:00
return R_FALSE;
if (from == to)
to = from+1;
if (type == 100) {
if ((to-from)<3) {
return R_FALSE;
}
}
2014-03-04 11:23:52 +00:00
/* set entry */
e_str = sdb_encode ((const void*)str, -1);
2014-03-04 11:23:52 +00:00
snprintf (key, sizeof (key)-1, "meta.%c.0x%"PFMT64x, type, from);
snprintf (val, sizeof (val)-1, "%d,%s", (int)(to-from), e_str);
exists = sdb_exists (DB, key);
2014-03-04 11:23:52 +00:00
sdb_set (DB, key, val, 0);
free (e_str);
// XXX: This is totally inefficient, using array_add withuot
// checking return value is wrong practice, also it may lead
// to inconsistent DB, and pretty bad performance. We should
// store this list in a different storage that doesnt have
// those limits and it's O(1) instead of O(n)
if (!exists) {
/* set type index */
snprintf (key, sizeof (key)-1, "meta.0x%"PFMT64x, from);
snprintf (val, sizeof (val)-1, "%c", type);
sdb_array_add (DB, key, val, 0);
/* set type index */
snprintf (key, sizeof (key)-1, "meta.%c", type);
sdb_array_add_num (DB, key, from, 0);
}
2014-03-04 11:23:52 +00:00
return R_TRUE;
}
R_API RAnalMetaItem *r_meta_find(RAnal *a, ut64 off, int type, int where) {
2014-03-04 11:23:52 +00:00
static RAnalMetaItem it = {0};
// XXX: return allocated item? wtf
if (where != R_META_WHERE_HERE) {
eprintf ("THIS WAS NOT SUPOSED TO HAPPEN\n");
return NULL;
}
//char *range = get_in_range (off);
if (type == R_META_TYPE_ANY) {
2014-07-02 15:14:32 +00:00
const char *p;
char key [100];
2014-03-04 11:23:52 +00:00
snprintf (key, sizeof (key)-1, "meta.0x%"PFMT64x, off);
p = sdb_const_get (DB, key, 0);
// XXX: TODO unimplemented. see core/disasm.c:1070
2014-03-04 11:23:52 +00:00
} else {
// snprintf (key, sizeof (key)-1, "meta.
}
return &it;
}
R_API const char *r_meta_type_to_string(int type) {
// XXX: use type as '%c'
switch(type) {
case R_META_TYPE_HIDE: return "Ch";
case R_META_TYPE_CODE: return "Cc";
case R_META_TYPE_DATA: return "Cd";
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_COMMENT: return "CC";
}
return "(...)";
}
static void printmetaitem(RAnal *a, RAnalMetaItem *d, int rad) {
char *pstr, *str = r_str_escape (d->str);
if (str) {
2014-04-29 00:46:36 +00:00
if (d->type=='s' && !*str) {
free (str);
return;
2014-04-29 00:46:36 +00:00
}
if (d->type != 'C') {
r_name_filter (str, 0);
pstr = str;
} else pstr = d->str;
// r_str_sanitize (str);
switch (rad) {
case 'j':
a->printf ("{\"offset\":%"PFMT64d", \"type\":\"%s\", \"name\":\"%s\"}",
d->from, r_meta_type_to_string (d->type), str);
break;
case 0:
a->printf ("0x%08"PFMT64x" %s\n",
d->from, str);
case 1:
case '*':
default:
2014-01-20 00:42:16 +00:00
if (d->type == 'C') {
a->printf ("\"%s %s\" @ 0x%08"PFMT64x"\n",
2014-01-20 00:42:16 +00:00
r_meta_type_to_string (d->type), pstr, d->from);
} else {
a->printf ("%s %d 0x%08"PFMT64x" # %s\n",
r_meta_type_to_string (d->type),
d->size, d->from, pstr);
2014-01-20 00:42:16 +00:00
}
break;
}
free (str);
}
}
2014-03-04 11:23:52 +00:00
typedef struct {
RAnal *anal;
int type;
int rad;
} RAnalMetaUserItem;
static int meta_print_item(void *user, const char *k, const char *v) {
RAnalMetaUserItem *ui = user;
RAnalMetaItem it;
if (strlen (k)<8)
return 1;
2014-03-04 11:23:52 +00:00
if (k[6]!='.')
return 1;
2014-03-04 11:23:52 +00:00
it.type = k[5];
it.size = sdb_atoi (v);
it.from = sdb_atoi (k+7);
it.to = it.from + it.size;
it.str = strchr (v, ',');
if (it.str)
it.str = (char *)sdb_decode ((const char*)it.str+1, 0);
printmetaitem (ui->anal, &it, ui->rad);
free (it.str);
return 1;
2014-03-04 11:23:52 +00:00
}
// TODO: Deprecate
R_API int r_meta_list(RAnal *a, int type, int rad) {
RAnalMetaUserItem ui = { a, type, rad };
if (rad=='j') a->printf ("[");
2014-03-04 11:23:52 +00:00
sdb_foreach (DB, meta_print_item, &ui);
if (rad=='j') a->printf ("]\n");
2014-03-04 11:23:52 +00:00
return 0;
}
#if 0
R_API char *r_anal_meta_bar (RAnal *anal, ut64 from, ut64 to, int blocks) {
int i, n, blocksize;
char *res;
ut64 f, t;
if (blocks<1 || from > to)
return NULL;
blocksize = (to-from)/blocks;
2014-03-04 11:23:52 +00:00
res = malloc (blocks*4);
for (i=0; i< blocks; i++) {
f = from + (blocksize*i);
t = f+blocksize;
n = r_anal_fcn_count (anal, f, t);
if (n>0) res[i++] = 'f';
res[i++] = ',';
}
return res;
}
#endif