2011-06-08 23:20:02 +00:00
|
|
|
/* radare - LGPL - Copyright 2008-2011 pancake<nopcode.org> */
|
2010-03-12 18:11:43 +00:00
|
|
|
|
|
|
|
#include <r_debug.h>
|
|
|
|
|
2010-06-03 22:56:44 +00:00
|
|
|
R_API RDebugTrace *r_debug_trace_new () {
|
|
|
|
RDebugTrace *t = R_NEW (RDebugTrace);
|
2010-06-04 21:47:35 +00:00
|
|
|
t->tag = 1; // UT32_MAX;
|
2010-06-13 09:59:17 +00:00
|
|
|
t->addresses = NULL;
|
2010-06-03 22:56:44 +00:00
|
|
|
t->enabled = R_FALSE;
|
|
|
|
t->traces = r_list_new ();
|
|
|
|
t->traces->free = free;
|
|
|
|
return t;
|
2010-03-12 18:11:43 +00:00
|
|
|
}
|
|
|
|
|
2010-06-03 22:56:44 +00:00
|
|
|
R_API void r_debug_trace_free (RDebug *dbg) {
|
|
|
|
if (dbg->trace == NULL)
|
|
|
|
return;
|
|
|
|
r_list_destroy (dbg->trace->traces);
|
2013-11-17 10:25:45 +00:00
|
|
|
free (dbg->trace->traces);
|
2010-06-03 22:56:44 +00:00
|
|
|
free (dbg->trace);
|
|
|
|
dbg->trace = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: added overlap/mask support here... wtf?
|
|
|
|
// TODO: think about tagged traces
|
|
|
|
R_API int r_debug_trace_tag (RDebug *dbg, int tag) {
|
|
|
|
//if (tag>0 && tag<31) core->dbg->trace->tag = 1<<(sz-1);
|
|
|
|
return (dbg->trace->tag = (tag>0)? tag: UT32_MAX);
|
2010-03-12 18:11:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_debug_trace_pc (RDebug *dbg) {
|
2010-06-04 21:47:35 +00:00
|
|
|
ut8 buf[32];
|
2010-09-18 00:51:17 +00:00
|
|
|
RRegItem *ri;
|
2011-02-24 13:06:49 +00:00
|
|
|
RAnalOp op;
|
2010-07-15 22:02:33 +00:00
|
|
|
static ut64 oldpc = 0LL; // Must trace the previously traced instruction
|
2010-03-12 18:11:43 +00:00
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_FALSE);
|
2010-06-04 21:47:35 +00:00
|
|
|
if ((ri = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_PC], -1))) {
|
2010-03-12 18:11:43 +00:00
|
|
|
ut64 addr = r_reg_get_value (dbg->reg, ri);
|
2012-11-05 01:00:34 +00:00
|
|
|
if (addr)
|
2010-06-04 21:47:35 +00:00
|
|
|
if (dbg->iob.read_at (dbg->iob.io, addr, buf, sizeof (buf))>0) {
|
2011-02-24 13:06:49 +00:00
|
|
|
if (r_anal_op (dbg->anal, &op, addr, buf, sizeof (buf))>0) {
|
2010-07-15 22:02:33 +00:00
|
|
|
if (oldpc!=0LL)
|
2013-12-06 04:18:57 +00:00
|
|
|
r_debug_trace_add (dbg, oldpc, op.size);
|
2010-07-15 22:02:33 +00:00
|
|
|
oldpc = addr;
|
2010-06-04 21:47:35 +00:00
|
|
|
return R_TRUE;
|
|
|
|
} else eprintf ("trace_pc: cannot get opcode size at 0x%"PFMT64x"\n", addr);
|
2011-10-09 02:15:32 +00:00
|
|
|
} //else eprintf ("trace_pc: cannot read memory at 0x%"PFMT64x"\n", addr);
|
2010-06-04 21:47:35 +00:00
|
|
|
} else eprintf ("trace_pc: cannot get program counter\n");
|
2010-03-12 18:11:43 +00:00
|
|
|
return R_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-06-13 09:59:17 +00:00
|
|
|
R_API void r_debug_trace_at(RDebug *dbg, const char *str) {
|
|
|
|
// TODO: parse offsets and so use ut64 instead of strstr()
|
|
|
|
free (dbg->trace->addresses);
|
|
|
|
dbg->trace->addresses = (str&&*str)? strdup (str): NULL;
|
|
|
|
}
|
|
|
|
|
2010-06-04 21:47:35 +00:00
|
|
|
R_API RDebugTracepoint *r_debug_trace_get (RDebug *dbg, ut64 addr) {
|
2010-03-12 18:11:43 +00:00
|
|
|
/* TODO: handle opcode size .. warn when jumping in the middle of instructions */
|
2010-06-04 21:47:35 +00:00
|
|
|
int tag = dbg->trace->tag;
|
2012-02-14 17:19:16 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RDebugTracepoint *trace;
|
|
|
|
r_list_foreach (dbg->trace->traces, iter, trace) {
|
2010-06-03 22:56:44 +00:00
|
|
|
if (tag != 0 && !(dbg->trace->tag & (1<<tag)))
|
2010-03-12 18:11:43 +00:00
|
|
|
continue;
|
|
|
|
if (trace->addr == addr)
|
|
|
|
return trace;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-06-04 21:47:35 +00:00
|
|
|
R_API void r_debug_trace_list (RDebug *dbg, int mode) {
|
|
|
|
int tag = dbg->trace->tag;
|
2012-02-14 17:19:16 +00:00
|
|
|
RListIter *iter;
|
|
|
|
RDebugTracepoint *trace;
|
|
|
|
r_list_foreach (dbg->trace->traces, iter, trace) {
|
2010-06-04 21:47:35 +00:00
|
|
|
if (!trace->tag || (tag & trace->tag)) {
|
|
|
|
if (mode == 1)
|
|
|
|
dbg->printf ("at+ 0x%"PFMT64x" %d\n", trace->addr, trace->times);
|
|
|
|
else dbg->printf ("0x%08"PFMT64x" size=%d count=%d times=%d tag=%d\n",
|
|
|
|
trace->addr, trace->size, trace->count, trace->times, trace->tag);
|
|
|
|
}
|
2010-03-12 18:11:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-13 09:59:17 +00:00
|
|
|
// XXX: find better name, make it public?
|
|
|
|
static int r_debug_trace_is_traceable(RDebug *dbg, ut64 addr) {
|
|
|
|
int ret = R_TRUE;
|
|
|
|
char addr_str[32];
|
|
|
|
if (dbg->trace->addresses) {
|
|
|
|
snprintf (addr_str, sizeof (addr_str), "0x%08"PFMT64x, addr);
|
2010-06-18 09:09:19 +00:00
|
|
|
if (!strstr (dbg->trace->addresses, addr_str))
|
|
|
|
ret = R_FALSE;
|
2010-06-13 09:59:17 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-03-12 18:11:43 +00:00
|
|
|
/* sort insert, or separated sort function ? */
|
|
|
|
/* TODO: detect if inner opcode */
|
2010-06-04 21:47:35 +00:00
|
|
|
R_API RDebugTracepoint *r_debug_trace_add (RDebug *dbg, ut64 addr, int size) {
|
2010-06-13 09:59:17 +00:00
|
|
|
RDebugTracepoint *tp;
|
2010-06-04 21:47:35 +00:00
|
|
|
int tag = dbg->trace->tag;
|
2010-06-13 09:59:17 +00:00
|
|
|
if (!r_debug_trace_is_traceable (dbg, addr))
|
|
|
|
return NULL;
|
2011-02-11 10:22:43 +00:00
|
|
|
r_anal_trace_bb (dbg->anal, addr);
|
2010-06-13 09:59:17 +00:00
|
|
|
tp = r_debug_trace_get (dbg, addr);
|
2010-06-03 22:56:44 +00:00
|
|
|
if (!tp) {
|
|
|
|
tp = R_NEW (RDebugTracepoint);
|
2010-06-04 21:47:35 +00:00
|
|
|
memset (tp, 0, sizeof (RDebugTracepoint));
|
2010-06-03 22:56:44 +00:00
|
|
|
tp->stamp = r_sys_now ();
|
|
|
|
tp->addr = addr;
|
|
|
|
tp->tags = tag;
|
|
|
|
tp->size = size;
|
2010-06-04 21:47:35 +00:00
|
|
|
tp->count = dbg->trace->count++;
|
2010-06-17 00:22:50 +00:00
|
|
|
tp->times = 1;
|
2010-06-03 22:56:44 +00:00
|
|
|
r_list_append (dbg->trace->traces, tp);
|
2010-06-04 21:47:35 +00:00
|
|
|
} else tp->times++;
|
2010-06-03 22:56:44 +00:00
|
|
|
return tp;
|
2010-03-12 18:11:43 +00:00
|
|
|
}
|
2010-11-24 22:19:17 +00:00
|
|
|
|
|
|
|
R_API void r_debug_trace_reset (RDebug *dbg) {
|
|
|
|
RDebugTrace *t = dbg->trace;
|
|
|
|
r_list_destroy (t->traces);
|
|
|
|
t->traces = r_list_new ();
|
|
|
|
t->traces->free = free;
|
|
|
|
}
|