mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-20 20:55:32 +00:00
d2233a896b
- Implemented in r_debug_esil_* APIs - Support for step-into and continue - Expressions can be regs, numbers, math operations, ranges, comparisons.. - Honor dbg.prestep for stepping before or after analysing - read-write-exec watchpoints using native debug backend and esil emulation - OSX/iOS debugger error message is now nicer - anal/rpnesil renamed to esil - Add esi,edi in osx regprofile - Add r_cons_is_breaked() for consistency
232 lines
5.7 KiB
C
232 lines
5.7 KiB
C
/* radare - LGPL - Copyright 2009-2015 - 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) {
|
|
int i, size;
|
|
if (!dbg || !dbg->reg || !dbg->h)
|
|
return R_FALSE;
|
|
|
|
// Theres no point in syncing a dead target
|
|
if (r_debug_is_dead (dbg))
|
|
return R_FALSE;
|
|
|
|
// Check if the functions needed are available
|
|
if (write && !dbg->h->reg_write)
|
|
return R_FALSE;
|
|
if (!write && !dbg->h->reg_read)
|
|
return R_FALSE;
|
|
|
|
// Sync all the types sequentially if asked
|
|
i = (type == R_REG_TYPE_ALL) ? R_REG_TYPE_GPR : type;
|
|
|
|
do {
|
|
if (write) {
|
|
ut8 *buf = r_reg_get_bytes (dbg->reg, i, &size);
|
|
if (!buf || !dbg->h->reg_write (dbg, i, buf, size)) {
|
|
if (i==0)
|
|
eprintf ("r_debug_reg: error writing registers %d to %d\n", i, dbg->pid);
|
|
return R_FALSE;
|
|
}
|
|
} else {
|
|
//int bufsize = R_MAX (1024, dbg->reg->size*2); // i know. its hacky
|
|
int bufsize = dbg->reg->size;
|
|
ut8 *buf = malloc (bufsize);
|
|
if (dbg->h && dbg->h->reg_read) {
|
|
size = dbg->h->reg_read (dbg, i, buf, dbg->reg->size);
|
|
} else size = -1;
|
|
if (size < 0) {
|
|
eprintf ("r_debug_reg: error reading registers\n");
|
|
return R_FALSE;
|
|
}
|
|
if (size)
|
|
r_reg_set_bytes (dbg->reg, i, buf, R_MIN(size, bufsize));
|
|
free (buf);
|
|
}
|
|
// DO NOT BREAK R_REG_TYPE_ALL PLEASE
|
|
// break;
|
|
|
|
// Continue the syncronization or just stop if it was asked only for a single type of regs
|
|
} while ((type==R_REG_TYPE_ALL) && (i++ < R_REG_TYPE_LAST));
|
|
return R_TRUE;
|
|
}
|
|
|
|
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 = "%4s 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 '-':
|
|
dbg->printf ("f-%s\n", item->name);
|
|
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(RDebug *dbg, const char *name) {
|
|
// ignores errors
|
|
return r_debug_reg_get_err(dbg, name, NULL);
|
|
}
|
|
|
|
R_API ut64 r_debug_reg_get_err(RDebug *dbg, const char *name, int *err) {
|
|
RRegItem *ri = NULL;
|
|
ut64 ret = 0LL;
|
|
int role = r_reg_get_name_idx (name);
|
|
const char *pname = name;
|
|
if (err) *err = 0;
|
|
if (!dbg || !dbg->reg) {
|
|
if (err) *err = 1;
|
|
return UT64_MAX;
|
|
}
|
|
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);
|
|
if (err) *err = 1;
|
|
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;
|
|
}
|
|
|
|
// XXX: dup for get_Err!
|
|
R_API ut64 r_debug_num_callback(RNum *userptr, const char *str, int *ok) {
|
|
RDebug *dbg = (RDebug *)userptr;
|
|
// resolve using regnu
|
|
return r_debug_reg_get_err (dbg, str, ok);
|
|
}
|