2010-02-05 11:21:37 +00:00
|
|
|
/* radare - LGPL - Copyright 2009-2010 pancake<nopcode.org> */
|
2009-09-18 18:11:42 +00:00
|
|
|
|
|
|
|
#include <r_reg.h>
|
|
|
|
|
|
|
|
/* non-endian safe - used for raw mapping with system registers */
|
2010-09-18 00:51:17 +00:00
|
|
|
R_API ut8* r_reg_get_bytes(RReg *reg, int type, int *size) {
|
|
|
|
RRegArena *arena;
|
2010-02-05 11:21:37 +00:00
|
|
|
int i, sz, osize;
|
|
|
|
ut8 *buf;
|
2009-09-18 18:11:42 +00:00
|
|
|
if (type == -1) {
|
|
|
|
/* serialize ALL register types in a single buffer */
|
2009-09-19 21:41:25 +00:00
|
|
|
// owned buffer is returned
|
2010-02-05 11:21:37 +00:00
|
|
|
osize = sz = 0;
|
|
|
|
buf = malloc (8);
|
|
|
|
for (i=0; i<R_REG_TYPE_LAST; i++) {
|
|
|
|
arena = reg->regset[i].arena;
|
2009-09-19 21:41:25 +00:00
|
|
|
sz += arena->size;
|
2010-02-05 11:21:37 +00:00
|
|
|
buf = realloc (buf, sz);
|
|
|
|
memcpy (buf+osize, arena->bytes, arena->size);
|
|
|
|
osize += sz;
|
2009-09-19 21:41:25 +00:00
|
|
|
}
|
|
|
|
if (size)
|
|
|
|
*size = sz;
|
|
|
|
return buf;
|
2009-09-18 18:11:42 +00:00
|
|
|
}
|
|
|
|
|
2010-02-02 10:09:52 +00:00
|
|
|
if (type<0 || type>R_REG_TYPE_LAST)
|
2009-09-18 18:11:42 +00:00
|
|
|
return NULL;
|
2010-02-05 11:21:37 +00:00
|
|
|
sz = reg->regset[type].arena->size;
|
2009-09-18 18:11:42 +00:00
|
|
|
if (size)
|
2010-02-05 11:21:37 +00:00
|
|
|
*size = sz;
|
|
|
|
buf = malloc (sz);
|
|
|
|
if (buf == NULL)
|
|
|
|
return NULL;
|
|
|
|
memcpy (buf, reg->regset[type].arena->bytes, sz);
|
|
|
|
return buf;
|
2009-09-18 18:11:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* reduce number of return statements */
|
2010-09-18 00:51:17 +00:00
|
|
|
R_API int r_reg_set_bytes(RReg *reg, int type, const ut8* buf, int len) {
|
2009-09-19 21:41:25 +00:00
|
|
|
int i, ret = R_FALSE;
|
2009-09-18 18:11:42 +00:00
|
|
|
struct r_reg_set_t *regset;
|
2010-09-18 00:51:17 +00:00
|
|
|
RRegArena *arena;
|
2009-09-19 21:41:25 +00:00
|
|
|
int off = 0;
|
2009-09-18 18:11:42 +00:00
|
|
|
|
|
|
|
if (type == -1) {
|
2009-09-19 21:41:25 +00:00
|
|
|
ret = R_TRUE;
|
2009-09-18 18:11:42 +00:00
|
|
|
/* deserialize ALL register types in a single buffer */
|
2010-02-05 11:21:37 +00:00
|
|
|
for(i=0; i<R_REG_TYPE_LAST; i++) {
|
2010-09-18 00:51:17 +00:00
|
|
|
if (!reg->regset[i].arena) {
|
|
|
|
arena = reg->regset[i].arena = R_NEW (RRegArena);
|
2009-09-25 02:04:51 +00:00
|
|
|
arena->size = len;
|
|
|
|
arena->bytes = malloc(len);
|
2010-09-18 00:51:17 +00:00
|
|
|
} else arena = reg->regset[i].arena;
|
2010-02-05 11:21:37 +00:00
|
|
|
if (arena->bytes == NULL)
|
|
|
|
return R_FALSE;
|
|
|
|
memcpy (arena->bytes, buf+off, arena->size);
|
2009-09-19 21:41:25 +00:00
|
|
|
off += arena->size;
|
|
|
|
if (off>len) {
|
|
|
|
ret = R_FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-09-18 18:11:42 +00:00
|
|
|
} else {
|
2010-02-02 10:09:52 +00:00
|
|
|
if (type>=0 && type<=R_REG_TYPE_LAST) {
|
2009-09-18 18:11:42 +00:00
|
|
|
regset = ®->regset[type];
|
2010-02-02 10:09:52 +00:00
|
|
|
if (len<=regset->arena->size) {
|
2010-01-19 10:25:17 +00:00
|
|
|
memcpy (regset->arena->bytes, buf, len);
|
2009-09-18 18:11:42 +00:00
|
|
|
ret = R_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2009-09-19 19:54:22 +00:00
|
|
|
|
2010-09-18 00:51:17 +00:00
|
|
|
R_API int r_reg_export_to(RReg *reg, RReg *dst) {
|
|
|
|
RRegItem *r;
|
|
|
|
RListIter *iter;
|
2009-09-19 21:41:25 +00:00
|
|
|
int i, ret = R_FALSE;
|
2009-09-19 19:54:22 +00:00
|
|
|
if (dst) {
|
|
|
|
// foreach reg of every time in reg, define it in dst
|
2009-09-19 21:41:25 +00:00
|
|
|
for(i=0;i<R_REG_TYPE_LAST;i++) {
|
2010-09-18 00:51:17 +00:00
|
|
|
r_list_foreach (reg->regset[i].regs, iter, r) {
|
2010-02-05 11:21:37 +00:00
|
|
|
// TODO: export to not implemented
|
2009-09-19 21:41:25 +00:00
|
|
|
//r_reg_set(dst, r_reg_get(dst, r->name), );
|
|
|
|
//r_mem_copybits_delta(
|
|
|
|
}
|
|
|
|
}
|
2009-09-19 19:54:22 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-09-18 00:51:17 +00:00
|
|
|
R_API int r_reg_fit_arena(RReg *reg) {
|
|
|
|
RRegArena *arena;
|
|
|
|
RListIter *iter;
|
|
|
|
RRegItem *r;
|
2009-09-19 19:54:22 +00:00
|
|
|
int size, i;
|
|
|
|
|
|
|
|
/* propagate arenas */
|
2010-09-18 00:51:17 +00:00
|
|
|
for (i=0; i<R_REG_TYPE_LAST; i++) {
|
2009-09-19 19:54:22 +00:00
|
|
|
arena = reg->regset[i].arena;
|
|
|
|
arena->size = 0;
|
2010-09-18 00:51:17 +00:00
|
|
|
r_list_foreach (reg->regset[i].regs, iter, r) {
|
|
|
|
size = BITS2BYTES (r->offset+r->size);
|
2009-09-19 19:54:22 +00:00
|
|
|
if (size>arena->size) {
|
|
|
|
arena->size = size;
|
2010-02-05 11:21:37 +00:00
|
|
|
arena->bytes = realloc (arena->bytes, size);
|
|
|
|
if (arena->bytes == NULL)
|
|
|
|
return R_FALSE;
|
2009-09-19 19:54:22 +00:00
|
|
|
}
|
|
|
|
}
|
2010-09-18 00:51:17 +00:00
|
|
|
memset (arena->bytes, 0, arena->size);
|
2009-09-19 19:54:22 +00:00
|
|
|
}
|
2010-02-05 11:21:37 +00:00
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-09-18 00:51:17 +00:00
|
|
|
R_API RRegArena *r_reg_arena_new (int size) {
|
|
|
|
RRegArena *arena = R_NEW (RRegArena);
|
2010-02-05 11:21:37 +00:00
|
|
|
if (arena) {
|
2010-09-18 00:51:17 +00:00
|
|
|
if (size<1)
|
|
|
|
size = 1;
|
|
|
|
if (!(arena->bytes = malloc (size+8))) {
|
2010-02-05 11:21:37 +00:00
|
|
|
free (arena);
|
|
|
|
arena = NULL;
|
|
|
|
} else arena->size = size;
|
|
|
|
}
|
|
|
|
return arena;
|
2009-09-19 19:54:22 +00:00
|
|
|
}
|
2010-09-18 00:51:17 +00:00
|
|
|
|
|
|
|
R_API void r_reg_arena_free(RRegArena* ra) {
|
|
|
|
free (ra->bytes);
|
|
|
|
free (ra);
|
|
|
|
}
|
2010-09-23 18:42:35 +00:00
|
|
|
|
|
|
|
R_API void r_reg_arena_swap(RReg *reg, int copy) {
|
|
|
|
r_reg_arena_set (reg, (++reg->iters)%2, copy);
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_reg_arena_set(RReg *reg, int n, int copy) {
|
|
|
|
int i;
|
|
|
|
if (n>r_list_length (reg->regset[0].pool))
|
|
|
|
return R_FALSE;
|
|
|
|
for (i=0; i<R_REG_TYPE_LAST; i++) {
|
|
|
|
RRegArena *o = reg->regset[i].arena;
|
|
|
|
RRegArena *a = (RRegArena*)r_list_get_n (reg->regset[i].pool, n);
|
|
|
|
reg->regset[i].arena = a;
|
|
|
|
if (a->size != o->size) {
|
|
|
|
a->size = o->size;
|
|
|
|
a->bytes = realloc (a->bytes, a->size+1);
|
|
|
|
if (!a->bytes) {
|
|
|
|
eprintf ("Cannot malloc %d in arena\n", a->size);
|
|
|
|
// XXX This is critical!
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (copy)
|
|
|
|
r_reg_set_bytes(reg, i, o->bytes, a->size);
|
|
|
|
}
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_reg_arena_pop(RReg *reg) {
|
|
|
|
int i;
|
|
|
|
for (i=0; i<R_REG_TYPE_LAST; i++) {
|
|
|
|
if (r_list_length(reg->regset[i].pool)>0) {
|
|
|
|
r_list_delete (reg->regset[i].pool,
|
|
|
|
r_list_head (reg->regset[i].pool));
|
|
|
|
// SEGFAULT: r_reg_arena_free (reg->regset[i].arena);
|
|
|
|
reg->regset[i].arena = (RRegArena*)r_list_head (
|
|
|
|
reg->regset[i].pool);
|
|
|
|
} else {
|
|
|
|
eprintf ("Cannot pop more\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_reg_arena_push(RReg *reg) {
|
|
|
|
int i;
|
|
|
|
for (i=0; i<R_REG_TYPE_LAST; i++) {
|
|
|
|
if (!(reg->regset[i].arena = r_reg_arena_new (0)))
|
|
|
|
return 0;
|
|
|
|
r_list_prepend (reg->regset[i].pool, reg->regset[i].arena);
|
|
|
|
}
|
|
|
|
return r_list_length (reg->regset[0].pool);
|
|
|
|
}
|