radare2/libr/anal/fcnstore.c
2017-02-24 23:42:17 +01:00

126 lines
2.9 KiB
C

/* radare - LGPL - Copyright 2011 -- pancake<nopcode.org> */
/* this file contains a test implementation of the ~O(1) function search */
// TODO: REFACTOR: This must be a generic data structure named RListRange
// TODO: We need a standard struct named Surface1D {.addr, .size}, so we can
// simplify all this by just passing the offset of the field of the given ptr
// TODO: RListComparator does not supports *user
// TODO: RRef - reference counting
#define RANGEBITS 10
// 1024
#define RANGE (1<<RANGEBITS)
#include <r_anal.h>
#if USE_NEW_FCN_STORE
#define STROFF(x) sdb_fmt (4, "flg.%"PFMT64x, x)
static int cmpfun(void *a, void *b) {
RAnalFunction *fa = (RAnalFunction*)a;
RAnalFunction *fb = (RAnalFunction*)b;
// TODO: swap sort order here or wtf?
return (fb->addr - fa->addr);
}
static void listrange_free_kv(HtKv *kv) {
free (kv->key);
r_list_free (kv->value);
free (kv);
}
R_API RListRange* r_listrange_new () {
RListRange *s = R_NEW (RListRange);
s->h = ht_new (NULL, listrange_free_kv, NULL);
//s->l shouldn't free is a helper structure to get data in order
//s->h will free the list that contains RAnalFunction
s->l = r_list_new ();
return s;
}
static inline ut64 r_listrange_next(ut64 addr) {
return (addr + RANGE);
}
R_API void r_listrange_free(RListRange *s) {
if (!s) {
return;
}
ht_free (s->h);
r_list_free (s->l);
free (s);
}
R_API void r_listrange_add(RListRange *s, RAnalFunction *f) {
ut64 addr;
RList *list;
ut64 from = f->addr;
ut64 to = f->addr + f->size;
for (addr = from; addr < to; addr = r_listrange_next (addr)) {
char *key = STROFF (addr);
list = ht_find (s->h, key, NULL);
if (list) {
if (!r_list_contains (list, f)) {
r_list_append (list, f);
}
} else {
list = r_list_new ((RListFree)r_anal_fcn_free);
r_list_append (list, f);
ht_insert (s->h, key, list);
}
}
r_list_add_sorted (s->l, f, cmpfun);
}
R_API void r_listrange_del(RListRange *s, RAnalFunction *f) {
RList *list;
ut64 addr, from, to;
if (!f) return;
from = f->addr;
to = f->addr + f->size;
for (addr = from; addr < to; addr = r_listrange_next (addr)) {
list = ht_find (s->h, STROFF (addr), NULL);
if (list) {
r_list_delete_data (list, f);
}
}
r_list_delete_data (s->l, f);
}
R_API void r_listrange_resize(RListRange *s, RAnalFunction *f, int newsize) {
if (!f) {
return;
}
r_listrange_del (s, f);
f->size = newsize;
r_listrange_add (s, f);
}
R_API RAnalFunction *r_listrange_find_in_range(RListRange* s, ut64 addr) {
RAnalFunction *f;
RListIter *iter;
RList *list = ht_find (s->h, STROFF (addr), NULL);
if (list) {
r_list_foreach (list, iter, f) {
if (R_BETWEEN (f->addr, addr, f->addr + f->_size))
return f;
}
}
return NULL;
}
R_API RAnalFunction *r_listrange_find_root(RListRange* s, ut64 addr) {
RAnalFunction *f;
RListIter *iter;
RList *list = ht_find (s->h, STROFF (addr), NULL);
if (list) {
r_list_foreach (list, iter, f) {
if (addr == f->addr) {
return f;
}
}
}
return NULL;
}
#endif