mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-01 00:51:19 +00:00
613 lines
14 KiB
C
613 lines
14 KiB
C
/* radare2 - LGPL - Copyright 2017-2020 - condret */
|
|
|
|
#include <r_util.h>
|
|
#include <r_types.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#if R2__WINDOWS__
|
|
#include <search.h>
|
|
#endif
|
|
|
|
static ut32 get_msb(ut32 v) {
|
|
int i;
|
|
for (i = 31; i > (-1); i--) {
|
|
if (v & (0x1U << i)) {
|
|
return (v & (0x1U << i));
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
R_API RIDPool* r_id_pool_new(ut32 start_id, ut32 last_id) {
|
|
RIDPool* pool = NULL;
|
|
if (start_id < last_id) {
|
|
pool = R_NEW0 (RIDPool);
|
|
if (pool) {
|
|
pool->next_id = pool->start_id = start_id;
|
|
pool->last_id = last_id;
|
|
}
|
|
}
|
|
return pool;
|
|
}
|
|
|
|
R_API bool r_id_pool_grab_id(RIDPool* pool, ut32* grabber) {
|
|
r_return_val_if_fail (pool && grabber, false);
|
|
|
|
*grabber = UT32_MAX;
|
|
if (pool->freed_ids) {
|
|
ut32 grab = (ut32) (size_t)r_queue_dequeue (pool->freed_ids);
|
|
*grabber = (ut32) grab;
|
|
if (r_queue_is_empty (pool->freed_ids)) {
|
|
r_queue_free (pool->freed_ids);
|
|
pool->freed_ids = NULL;
|
|
}
|
|
return true;
|
|
}
|
|
if (pool->next_id < pool->last_id) {
|
|
*grabber = pool->next_id;
|
|
pool->next_id++;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_id_pool_kick_id(RIDPool* pool, ut32 kick) {
|
|
if (!pool || (kick < pool->start_id) || (pool->start_id == pool->next_id)) {
|
|
return false;
|
|
}
|
|
if (kick == (pool->next_id - 1)) {
|
|
pool->next_id--;
|
|
return true;
|
|
}
|
|
if (!pool->freed_ids) {
|
|
pool->freed_ids = r_queue_new (2);
|
|
}
|
|
r_queue_enqueue (pool->freed_ids, (void*) (size_t) kick);
|
|
return true;
|
|
}
|
|
|
|
R_API void r_id_pool_free(RIDPool* pool) {
|
|
if (pool && pool->freed_ids) {
|
|
r_queue_free (pool->freed_ids);
|
|
}
|
|
free (pool);
|
|
}
|
|
|
|
R_API RIDStorage* r_id_storage_new(ut32 start_id, ut32 last_id) {
|
|
RIDStorage* storage = NULL;
|
|
RIDPool *pool = r_id_pool_new (start_id, last_id);
|
|
if (pool) {
|
|
storage = R_NEW0 (RIDStorage);
|
|
if (!storage) {
|
|
r_id_pool_free (pool);
|
|
return NULL;
|
|
}
|
|
storage->pool = pool;
|
|
}
|
|
return storage;
|
|
}
|
|
|
|
static bool id_storage_reallocate(RIDStorage* storage, ut32 size) {
|
|
if (!storage) {
|
|
return false;
|
|
}
|
|
void **data = realloc (storage->data, size * sizeof (void*));
|
|
if (!data) {
|
|
return false;
|
|
}
|
|
if (size > storage->size) {
|
|
memset (data + storage->size, 0, (size - storage->size) * sizeof (void*));
|
|
}
|
|
storage->data = data;
|
|
storage->size = size;
|
|
return true;
|
|
}
|
|
|
|
static bool oid_storage_preallocate(ROIDStorage *st, ut32 size) {
|
|
ut32 *permutation;
|
|
if (!st) {
|
|
return false;
|
|
}
|
|
if (!size) {
|
|
R_FREE (st->permutation);
|
|
st->psize = 0;
|
|
}
|
|
permutation = realloc (st->permutation, size * sizeof (ut32));
|
|
if (!permutation) {
|
|
return false;
|
|
}
|
|
if (size > st->psize) {
|
|
memset (permutation + st->psize, 0, (size - st->psize) * sizeof (ut32));
|
|
}
|
|
st->permutation = permutation;
|
|
st->psize = size;
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_id_storage_set(RIDStorage* storage, void* data, ut32 id) {
|
|
ut32 n;
|
|
if (!storage || !storage->pool || (id >= storage->pool->next_id)) {
|
|
return false;
|
|
}
|
|
n = get_msb (id + 1);
|
|
if (n > ((storage->size / 2) + (storage->size / 4))) {
|
|
if ((n * 2) < storage->pool->last_id) {
|
|
if (!id_storage_reallocate (storage, n * 2)) {
|
|
return false;
|
|
}
|
|
} else if (n != (storage->pool->last_id)) {
|
|
if (!id_storage_reallocate (storage, storage->pool->last_id)) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
storage->data[id] = data;
|
|
if (id > storage->top_id) {
|
|
storage->top_id = id;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_id_storage_add(RIDStorage* storage, void* data, ut32* id) {
|
|
if (!storage || !r_id_pool_grab_id (storage->pool, id)) {
|
|
return false;
|
|
}
|
|
return r_id_storage_set (storage, data, *id);
|
|
}
|
|
|
|
R_API void* r_id_storage_get(RIDStorage* storage, ut32 id) {
|
|
if (!storage || !storage->data || (storage->size <= id)) {
|
|
return NULL;
|
|
}
|
|
return storage->data[id];
|
|
}
|
|
|
|
R_API bool r_id_storage_get_lowest(RIDStorage *storage, ut32 *id) {
|
|
r_return_val_if_fail (storage, false);
|
|
ut32 i;
|
|
for (i = 0; i < storage->size && !storage->data[i]; i++);
|
|
*id = i;
|
|
return i < storage->size;
|
|
}
|
|
|
|
R_API bool r_id_storage_get_highest(RIDStorage *storage, ut32 *id) {
|
|
r_return_val_if_fail (storage, false);
|
|
size_t i = 0;
|
|
if (storage->size > 0) {
|
|
for (i = storage->size - 1; !storage->data[i] && i > 0; i--);
|
|
*id = i;
|
|
return storage->data[i];
|
|
}
|
|
// *id = i;
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_id_storage_get_next(RIDStorage *storage, ut32 *idref) {
|
|
r_return_val_if_fail (idref && storage, false);
|
|
ut32 id = *idref;
|
|
if (storage->size < 1 || id >= storage->size || !storage->data) {
|
|
return false;
|
|
}
|
|
for (id = *idref + 1; id < storage->size && !storage->data[id]; id++);
|
|
if (id < storage->size) {
|
|
*idref = id;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_id_storage_get_prev(RIDStorage *storage, ut32 *idref) {
|
|
r_return_val_if_fail (idref && storage, false);
|
|
ut32 id = *idref;
|
|
if (id == 0 || id >= storage->size || storage->size < 1 || !storage->data) {
|
|
return false;
|
|
}
|
|
for (id = *idref - 1; id > 0 && !storage->data[id]; id--);
|
|
if (storage->data[id]) {
|
|
*idref = id;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API void r_id_storage_delete(RIDStorage* storage, ut32 id) {
|
|
if (!storage || !storage->data || (storage->size <= id)) {
|
|
return;
|
|
}
|
|
storage->data[id] = NULL;
|
|
if (id == storage->top_id) {
|
|
while (storage->top_id && !storage->data[storage->top_id]) {
|
|
storage->top_id--;
|
|
}
|
|
if (!storage->top_id) {
|
|
if (storage->data[storage->top_id]) {
|
|
id_storage_reallocate (storage, 2);
|
|
} else {
|
|
RIDPool* pool = r_id_pool_new (storage->pool->start_id, storage->pool->last_id);
|
|
R_FREE (storage->data);
|
|
storage->size = 0;
|
|
r_id_pool_free (storage->pool);
|
|
storage->pool = pool;
|
|
return;
|
|
}
|
|
} else if ((storage->top_id + 1) < (storage->size / 4)) {
|
|
id_storage_reallocate (storage, storage->size / 2);
|
|
}
|
|
}
|
|
r_id_pool_kick_id (storage->pool, id);
|
|
}
|
|
|
|
R_API void* r_id_storage_take(RIDStorage* storage, ut32 id) {
|
|
void* ret = r_id_storage_get (storage, id);
|
|
r_id_storage_delete (storage, id);
|
|
return ret;
|
|
}
|
|
|
|
R_API bool r_id_storage_foreach(RIDStorage* storage, RIDStorageForeachCb cb, void* user) {
|
|
ut32 i;
|
|
if (!cb || !storage || !storage->data) {
|
|
return false;
|
|
}
|
|
for (i = 0; i < storage->top_id; i++) {
|
|
if (storage->data[i] && !cb (user, storage->data[i], i)) {
|
|
return false;
|
|
}
|
|
}
|
|
if (storage->data[i]) {
|
|
return cb (user, storage->data[i], i);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API void r_id_storage_free(RIDStorage* storage) {
|
|
if (storage) {
|
|
r_id_pool_free (storage->pool);
|
|
free (storage->data);
|
|
}
|
|
free (storage);
|
|
}
|
|
|
|
static bool _list(void* user, void* data, ut32 id) {
|
|
r_list_append (user, data);
|
|
return true;
|
|
}
|
|
|
|
R_API RList *r_id_storage_list(RIDStorage *s) { //remove this pls
|
|
RList *list = r_list_newf (NULL);
|
|
r_id_storage_foreach (s, _list, list);
|
|
return list;
|
|
}
|
|
|
|
R_API ROIDStorage *r_oids_new(ut32 start_id, ut32 last_id) {
|
|
ROIDStorage *storage = R_NEW0 (ROIDStorage);
|
|
if (!storage) {
|
|
return NULL;
|
|
}
|
|
if (!(storage->data = r_id_storage_new (start_id, last_id))) {
|
|
free (storage);
|
|
return NULL;
|
|
}
|
|
return storage;
|
|
}
|
|
|
|
R_API void *r_oids_get(ROIDStorage *storage, ut32 id) {
|
|
if (storage) {
|
|
return r_id_storage_get (storage->data, id);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
R_API void *r_oids_oget(ROIDStorage *storage, ut32 od) {
|
|
ut32 id;
|
|
if (r_oids_get_id (storage, od, &id)) {
|
|
return r_id_storage_get (storage->data, id);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
R_API bool r_oids_get_id(ROIDStorage *storage, ut32 od, ut32 *id) {
|
|
if (storage && storage->permutation && (storage->ptop > od)) {
|
|
*id = storage->permutation[od];
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_oids_get_od(ROIDStorage *storage, ut32 id, ut32 *od) {
|
|
if (storage && storage->permutation &&
|
|
storage->data && (id < storage->data->pool->next_id)) {
|
|
for (od[0] = 0; od[0] < storage->ptop; od[0]++) {
|
|
if (id == storage->permutation[od[0]]) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API bool r_oids_add(ROIDStorage *storage, void *data, ut32 *id, ut32 *od) {
|
|
if (!storage || !id || !od) {
|
|
return false;
|
|
}
|
|
if (!r_id_storage_add (storage->data, data, id)) {
|
|
return false;
|
|
}
|
|
if (!storage->permutation) {
|
|
oid_storage_preallocate (storage, 4);
|
|
} else if (storage->ptop > (storage->psize * 3 / 4)) {
|
|
oid_storage_preallocate (storage, storage->psize * 2);
|
|
}
|
|
if (storage->psize <= storage->ptop) {
|
|
r_id_storage_delete (storage->data, *id);
|
|
return false;
|
|
}
|
|
if (!storage->permutation) {
|
|
return false;
|
|
}
|
|
*od = storage->ptop;
|
|
storage->permutation[*od] = *id;
|
|
storage->ptop++;
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_oids_to_front(ROIDStorage *storage, const ut32 id) {
|
|
ut32 od;
|
|
if (!storage || !storage->permutation) {
|
|
return false;
|
|
}
|
|
for (od = 0; od < storage->ptop; od++) {
|
|
if (id == storage->permutation[od]) {
|
|
break;
|
|
}
|
|
}
|
|
if (od == storage->ptop) {
|
|
return false;
|
|
} else if (od == (storage->ptop - 1)) {
|
|
return true;
|
|
}
|
|
memmove (&storage->permutation[od], &storage->permutation[od + 1],
|
|
(storage->ptop - od - 1) * sizeof (ut32));
|
|
storage->permutation[storage->ptop - 1]= id;
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_oids_to_rear(ROIDStorage *storage, ut32 id) {
|
|
ut32 od;
|
|
if (!storage || !storage->permutation ||
|
|
!storage->data || (id >= storage->data->pool->next_id)) {
|
|
return false;
|
|
}
|
|
bool found = false;
|
|
for (od = 0; od < storage->ptop; od++) {
|
|
if (id == storage->permutation[od]) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (od == storage->ptop) {
|
|
return false;
|
|
}
|
|
if (!found) {
|
|
return true;
|
|
}
|
|
memmove (&storage->permutation[1], &storage->permutation[0], od * sizeof (ut32));
|
|
storage->permutation[0] = id;
|
|
return true;
|
|
}
|
|
|
|
R_API void r_oids_delete(ROIDStorage *storage, ut32 id) {
|
|
if (!r_oids_to_front (storage, id)) {
|
|
return;
|
|
}
|
|
r_id_storage_delete (storage->data, id);
|
|
storage->ptop--;
|
|
if (!storage->ptop) {
|
|
R_FREE (storage->permutation);
|
|
storage->psize = 0;
|
|
} else if ((storage->ptop + 1) < (storage->psize / 4)) {
|
|
oid_storage_preallocate (storage, storage->psize / 2);
|
|
}
|
|
}
|
|
|
|
R_API void r_oids_odelete(ROIDStorage *st, ut32 od) {
|
|
ut32 n;
|
|
if (!st || !st->permutation || od >= st->ptop) {
|
|
return;
|
|
}
|
|
n = st->ptop - od - 1;
|
|
r_id_storage_delete (st->data, st->permutation[od]);
|
|
memmove (&st->permutation[od], &st->permutation[od + 1], n * sizeof (ut32));
|
|
st->ptop--;
|
|
if (!st->ptop) {
|
|
R_FREE (st->permutation);
|
|
st->psize = 0;
|
|
} else if ((st->ptop + 1) < (st->psize / 4)) {
|
|
oid_storage_preallocate (st, st->psize / 2);
|
|
}
|
|
}
|
|
|
|
R_API void *r_oids_take(ROIDStorage *storage, ut32 id) {
|
|
r_return_val_if_fail (storage, NULL);
|
|
void *ret = r_id_storage_get (storage->data, id);
|
|
r_oids_delete (storage, id);
|
|
return ret;
|
|
}
|
|
|
|
R_API void *r_oids_otake(ROIDStorage *st, ut32 od) {
|
|
void *ret = r_oids_oget (st, od);
|
|
r_oids_odelete (st, od);
|
|
return ret;
|
|
}
|
|
|
|
R_API void r_oids_free(ROIDStorage *storage) {
|
|
if (storage) {
|
|
free (storage->permutation);
|
|
r_id_storage_free (storage->data);
|
|
}
|
|
free (storage);
|
|
}
|
|
|
|
//returns the element with lowest order
|
|
R_API void *r_oids_last(ROIDStorage *storage) {
|
|
if (storage && storage->data && storage->data->data
|
|
&& storage->permutation) {
|
|
return storage->data->data[storage->permutation[0]];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
//return the element with highest order
|
|
R_API void *r_oids_first(ROIDStorage *storage) {
|
|
if (storage && storage->data && storage->data->data
|
|
&& storage->permutation) {
|
|
return storage->data->data[storage->permutation[storage->ptop - 1]];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
R_API bool r_oids_foreach(ROIDStorage *storage, RIDStorageForeachCb cb, void *user) {
|
|
ut32 i;
|
|
ut32 id;
|
|
if (!cb || !storage || !storage->data || !storage->data->data
|
|
|| !storage->permutation) {
|
|
return false;
|
|
}
|
|
for (i = storage->ptop - 1; i != 0; i--) {
|
|
id = storage->permutation[i];
|
|
if (!cb (user, storage->data->data[id], id)) {
|
|
return false;
|
|
}
|
|
}
|
|
id = storage->permutation[0];
|
|
return cb (user, storage->data->data[id], id);
|
|
}
|
|
|
|
R_API bool r_oids_foreach_prev(ROIDStorage* storage, RIDStorageForeachCb cb, void* user) {
|
|
ut32 i;
|
|
ut32 id;
|
|
if (!cb || !storage || !storage->data || !storage->data->data
|
|
|| !storage->permutation) {
|
|
return false;
|
|
}
|
|
for (i = 0; i < storage->ptop; i++) {
|
|
id = storage->permutation[i];
|
|
if (!cb (user, storage->data->data[id], id)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool oids_od_bfind (ROIDStorage *st, ut32 *od, void *incoming, void *user) {
|
|
int cmp_res;
|
|
void *in;
|
|
|
|
if (!st->ptop) {
|
|
return false;
|
|
}
|
|
|
|
st64 high = st->ptop - 1;
|
|
st64 low = 0;
|
|
|
|
while (1) {
|
|
if (high <= low) {
|
|
od[0] = (ut32)low;
|
|
in = r_oids_oget(st, od[0]);
|
|
//in - incoming
|
|
if (!st->cmp (in, incoming, user, &cmp_res)) {
|
|
return false;
|
|
}
|
|
if (cmp_res < 0) {
|
|
od[0]++;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
od[0] = (ut32)((low + high) / 2);
|
|
in = r_oids_oget (st, od[0]);
|
|
if (!st->cmp (in, incoming, user, &cmp_res)) {
|
|
return false;
|
|
}
|
|
|
|
if (cmp_res == 0) {
|
|
return true;
|
|
}
|
|
|
|
if (cmp_res < 0) {
|
|
low = od[0] + 1;
|
|
} else {
|
|
high = od[0];
|
|
high--;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool oids_od_binsert(ROIDStorage *storage, ut32 id, ut32 *od, void *incoming, void *user) {
|
|
if (!oids_od_bfind (storage, od, incoming, user)) {
|
|
return false;
|
|
}
|
|
if (od[0] != storage->ptop) {
|
|
memmove (&storage->permutation[od[0] + 1], &storage->permutation[od[0]], (storage->ptop - od[0]) * sizeof (ut32));
|
|
}
|
|
storage->ptop++;
|
|
storage->permutation[od[0]] = id;
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_oids_insert(ROIDStorage *storage, void *data, ut32 *id, ut32 *od, void *user) {
|
|
if (!storage || !storage->cmp || !id || !od) {
|
|
return false;
|
|
}
|
|
if (!storage->ptop) { //empty storage
|
|
return r_oids_add (storage, data, id, od);
|
|
}
|
|
if (!r_id_storage_add (storage->data, data, id)) {
|
|
return false;
|
|
}
|
|
if (storage->ptop > (storage->psize * 3 / 4)) {
|
|
oid_storage_preallocate (storage, storage->psize * 2);
|
|
}
|
|
return oids_od_binsert (storage, id[0], od, data, user);
|
|
}
|
|
|
|
R_API bool r_oids_sort(ROIDStorage *storage, void *user) {
|
|
ut32 od, id, ptop, *permutation;
|
|
|
|
if (!storage || !storage->ptop || !storage->cmp) {
|
|
return false;
|
|
}
|
|
if (storage->ptop == 1) {
|
|
return true;
|
|
}
|
|
permutation = storage->permutation;
|
|
storage->permutation = R_NEWS0 (ut32, storage->psize);
|
|
if (!storage->permutation) {
|
|
storage->permutation = permutation;
|
|
return false;
|
|
}
|
|
storage->permutation[0] = permutation[0];
|
|
ptop = storage->ptop;
|
|
storage->ptop = 1;
|
|
while (storage->ptop != ptop) {
|
|
id = permutation[storage->ptop];
|
|
void *incoming = r_id_storage_get (storage->data, id);
|
|
if (!oids_od_binsert (storage, id, &od, incoming, user)) {
|
|
goto beach;
|
|
}
|
|
}
|
|
free (permutation);
|
|
return true;
|
|
|
|
beach:
|
|
free (storage->permutation);
|
|
storage->permutation = permutation;
|
|
storage->ptop = ptop;
|
|
return false;
|
|
}
|
|
|
|
R_API ut32 r_oids_find(ROIDStorage *storage, void *incoming, void *user) {
|
|
ut32 ret;
|
|
return oids_od_bfind (storage, &ret, incoming, user) ? ret : storage->ptop;
|
|
}
|