2014-06-19 04:03:11 +02:00

204 lines
5.1 KiB
C

/* radare - LGPL - Copyright 2009-2014 - pancake */
#include <r_debug.h>
#include <r_cons.h>
#include <r_reg.h>
R_API int r_debug_reg_sync(RDebug *dbg, int type, int write) {
ut8 buf[4096]; // XXX hacky!
int next, size, ret = R_FALSE;
if (!dbg || !dbg->reg || dbg->pid == -1)
return R_FALSE;
if (type == -1) {
type = R_REG_TYPE_GPR;
next = R_REG_TYPE_DRX;
} else next = 0;
repeat:
if (write) {
if (dbg && dbg->h && dbg->h->reg_write) {
ut8 *buf = r_reg_get_bytes (dbg->reg, type, &size);
if (!dbg->h->reg_write (dbg, type, buf, sizeof (buf)))
eprintf ("r_debug_reg: error writing registers\n");
else ret = R_TRUE;
} //else eprintf ("r_debug_reg: cannot set registers\n");
} else {
/* read registers from debugger backend to dbg->regs */
if (dbg && dbg->h && dbg->h->reg_read) {
size = dbg->h->reg_read (dbg, type, buf, sizeof (buf));
if (size == 0) {
eprintf ("r_debug_reg: error reading registers pid=%d\n", dbg->pid);
} else {
ret = r_reg_set_bytes (dbg->reg, type, buf, size);
}
} //else eprintf ("r_debug_reg: cannot read registers\n");
}
if (next) {
type = next;
switch (next) {
case R_REG_TYPE_FPU: next = R_REG_TYPE_DRX; break;
case R_REG_TYPE_DRX: next = 0; break;
default: next = 0; break;
}
goto repeat;
}
return ret;
}
R_API int r_debug_reg_list(RDebug *dbg, int type, int size, int rad, const char *use_color) {
int i, delta, from, to, cols, n = 0;
const char *fmt, *fmt2, *kwhites;
RListIter *iter;
RRegItem *item;
RList *head;
ut64 diff;
if (!dbg || !dbg->reg)
return R_FALSE;
if (!(dbg->reg->bits & size)) {
// TODO: verify if 32bit exists, otherwise use 64 or 8?
size = 32;
}
//if (dbg->h && dbg->h->bits & R_SYS_BITS_64) {
if (dbg->bits & R_SYS_BITS_64) {
fmt = "%s = 0x%08"PFMT64x"%s";
fmt2 = "%4s 0x%08"PFMT64x"%s";
cols = 3;
kwhites = " ";
} else {
fmt = " %s = 0x%08"PFMT64x"%s";
fmt2 = " %3s 0x%08"PFMT64x"%s";
cols = 4;
kwhites = " ";
}
if (rad=='j')
dbg->printf ("{");
if (type == -1) {
from = 0;
to = R_REG_TYPE_LAST;
} else {
from = type;
to = from +1;
}
for (i=from; i<to; i++) {
head = r_reg_get_list (dbg->reg, i);
if (!head) continue;
r_list_foreach (head, iter, item) {
ut64 value;
if (type != -1) {
if (type != item->type)
continue;
if (size != 0 && size != item->size)
continue;
}
value = r_reg_get_value (dbg->reg, item);
r_reg_arena_swap (dbg->reg, R_FALSE);
diff = r_reg_get_value (dbg->reg, item);
r_reg_arena_swap (dbg->reg, R_FALSE);
delta = value-diff;
switch (rad) {
case 'j':
dbg->printf ("%s\"%s\":%"PFMT64d,
n?",":"", item->name, value);
break;
case 1:
case '*':
dbg->printf ("f %s 1 0x%"PFMT64x"\n",
item->name, value);
break;
case 'd':
case 2:
{
char *str, whites[16], content[128];
int len;
strcpy (whites, kwhites);
if (delta && use_color)
dbg->printf (use_color);
if (item->flags) {
str = r_reg_get_bvalue (dbg->reg, item);
len = strlen (str);
strcpy (whites, " ");
len = (len>9)?9:(9-len);
whites[len] = 0;
dbg->printf (" %s = %s%s", item->name,
str, ((n+1)%cols)? whites: "\n");
free (str);
} else {
snprintf (content, sizeof(content), fmt2, item->name, value, "");
len = strlen (content);
len -= 4;
if (len>10) {
len -= 10;
len = (len>9)?9:(9-len);
whites[len] = 0;
}
dbg->printf (fmt2, item->name, value,
((n+1)%cols)? whites: "\n");
}
if (delta && use_color)
dbg->printf (Color_RESET);
}
break;
case 3:
if (delta) {
char woot[64];
snprintf (woot, sizeof (woot),
" was 0x%08"PFMT64x" delta %d\n", diff, delta);
dbg->printf (fmt, item->name, value, woot);
}
break;
default:
if (delta && use_color)
dbg->printf (use_color);
dbg->printf (fmt, item->name, value, "\n");
if (delta && use_color)
dbg->printf (Color_RESET);
break;
}
n++;
}
}
if (rad=='j') dbg->printf ("}\n");
else if (n>0 && rad==2 && ((n%cols)))
dbg->printf ("\n");
return n;
}
R_API int r_debug_reg_set(struct r_debug_t *dbg, const char *name, ut64 num) {
RRegItem *ri;
int role = r_reg_get_name_idx (name);
if (!dbg || !dbg->reg)
return R_FALSE;
if (role != -1)
name = r_reg_get_name (dbg->reg, role);
ri = r_reg_get (dbg->reg, name, R_REG_TYPE_GPR);
if (ri) {
r_reg_set_value (dbg->reg, ri, num);
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_TRUE);
}
return (ri!=NULL);
}
R_API ut64 r_debug_reg_get(struct r_debug_t *dbg, const char *name) {
RRegItem *ri = NULL;
ut64 ret = 0LL;
int role = r_reg_get_name_idx (name);
const char *pname = name;
if (!dbg || !dbg->reg)
return R_FALSE;
if (role != -1) {
name = r_reg_get_name (dbg->reg, role);
if (name == NULL || *name == '\0') {
eprintf ("No debug register profile defined for '%s'.\n", pname);
return UT64_MAX;
}
}
ri = r_reg_get (dbg->reg, name, R_REG_TYPE_GPR);
if (ri) {
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_FALSE);
ret = r_reg_get_value (dbg->reg, ri);
}
return ret;
}