2015-01-30 02:07:47 +01:00
|
|
|
/* radare - LGPL - Copyright 2009-2015 - pancake, TheLemonMan */
|
2009-02-05 22:08:46 +01:00
|
|
|
|
|
|
|
#include <r_debug.h>
|
2010-03-12 18:46:11 +01:00
|
|
|
#include <signal.h>
|
|
|
|
|
2013-06-14 02:51:33 +02:00
|
|
|
R_LIB_VERSION(r_debug);
|
|
|
|
|
2014-09-17 14:57:33 +02:00
|
|
|
// Size of the lookahead buffers used in r_debug functions
|
|
|
|
#define DBG_BUF_SIZE 512
|
|
|
|
|
2014-06-04 04:12:11 +02:00
|
|
|
R_API RDebugInfo *r_debug_info(RDebug *dbg, const char *arg) {
|
2014-06-05 03:46:17 +02:00
|
|
|
if (!dbg || !dbg->h || !dbg->h->info)
|
2014-06-04 04:12:11 +02:00
|
|
|
return NULL;
|
|
|
|
return dbg->h->info (dbg, arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_debug_info_free (RDebugInfo *rdi) {
|
|
|
|
free (rdi->cwd);
|
|
|
|
free (rdi->exe);
|
|
|
|
free (rdi->cmdline);
|
2015-10-12 18:49:16 +02:00
|
|
|
free (rdi->libname);
|
2014-06-04 04:12:11 +02:00
|
|
|
}
|
|
|
|
|
2010-03-12 18:46:11 +01:00
|
|
|
/* restore program counter after breakpoint hit */
|
2010-05-24 12:07:54 +02:00
|
|
|
static int r_debug_recoil(RDebug *dbg) {
|
2011-05-20 20:42:25 +02:00
|
|
|
int recoil;
|
2010-09-18 02:51:17 +02:00
|
|
|
RRegItem *ri;
|
2011-05-20 20:42:25 +02:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, false);
|
2010-03-12 18:46:11 +01:00
|
|
|
ri = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_PC], -1);
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.bpi = NULL;
|
2010-03-12 18:46:11 +01:00
|
|
|
if (ri) {
|
|
|
|
ut64 addr = r_reg_get_value (dbg->reg, ri);
|
|
|
|
recoil = r_bp_recoil (dbg->bp, addr);
|
2013-09-18 02:11:23 +02:00
|
|
|
//eprintf ("[R2] Breakpoint recoil at 0x%"PFMT64x" = %d\n", addr, recoil);
|
2016-02-06 19:53:29 +01:00
|
|
|
if (recoil < 1)
|
|
|
|
recoil = 0; // XXX Hack :D
|
2010-03-12 18:46:11 +01:00
|
|
|
if (recoil) {
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.type = R_DEBUG_REASON_BREAKPOINT;
|
|
|
|
dbg->reason.bpi = r_bp_get_at (dbg->bp, addr-recoil);
|
|
|
|
dbg->reason.addr = addr - recoil;
|
2010-03-12 18:46:11 +01:00
|
|
|
r_reg_set_value (dbg->reg, ri, addr-recoil);
|
2015-08-17 00:12:54 +02:00
|
|
|
if (r_reg_get_value (dbg->reg, ri) != (addr-recoil)) {
|
2014-03-21 03:18:10 +01:00
|
|
|
eprintf ("r_debug_recoil: Cannot set program counter\n");
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-03-21 03:18:10 +01:00
|
|
|
}
|
2015-09-14 02:08:31 +02:00
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, true);
|
2011-07-06 01:45:45 +02:00
|
|
|
//eprintf ("[BP Hit] Setting pc to 0x%"PFMT64x"\n", (addr-recoil));
|
2015-09-14 02:08:31 +02:00
|
|
|
return true;
|
2015-06-09 00:47:13 +02:00
|
|
|
}
|
2015-08-29 14:01:38 +02:00
|
|
|
} else {
|
|
|
|
eprintf ("r_debug_recoil: Cannot get program counter\n");
|
|
|
|
}
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2010-03-12 18:46:11 +01:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
|
2010-05-24 12:07:54 +02:00
|
|
|
R_API RDebug *r_debug_new(int hard) {
|
2015-01-30 02:07:47 +01:00
|
|
|
RDebug *dbg = R_NEW0 (RDebug);
|
2015-07-21 06:06:00 +02:00
|
|
|
if (!dbg) return NULL;
|
|
|
|
// R_SYS_ARCH
|
2015-10-22 03:11:03 +02:00
|
|
|
dbg->arch = strdup (R_SYS_ARCH);
|
2015-07-21 06:06:00 +02:00
|
|
|
dbg->bits = R_SYS_BITS;
|
|
|
|
dbg->trace_forks = 1;
|
|
|
|
dbg->trace_clone = 0;
|
|
|
|
R_FREE (dbg->btalgo);
|
|
|
|
dbg->trace_execs = 0;
|
|
|
|
dbg->anal = NULL;
|
|
|
|
dbg->snaps = r_list_newf (r_debug_snap_free);
|
|
|
|
dbg->pid = -1;
|
|
|
|
dbg->bpsize = 1;
|
|
|
|
dbg->tid = -1;
|
|
|
|
dbg->tree = r_tree_new ();
|
|
|
|
dbg->tracenodes = sdb_new0 ();
|
|
|
|
dbg->swstep = 0;
|
|
|
|
dbg->newstate = 0;
|
2015-09-14 02:08:31 +02:00
|
|
|
dbg->stop_all_threads = false;
|
2015-07-21 06:06:00 +02:00
|
|
|
dbg->trace = r_debug_trace_new ();
|
2015-08-08 14:15:13 -04:00
|
|
|
dbg->cb_printf = (void *)printf;
|
2015-07-21 06:06:00 +02:00
|
|
|
dbg->reg = r_reg_new ();
|
|
|
|
dbg->num = r_num_new (r_debug_num_callback, dbg);
|
|
|
|
dbg->h = NULL;
|
2015-08-29 14:01:38 +02:00
|
|
|
dbg->threads = NULL;
|
2015-07-21 06:06:00 +02:00
|
|
|
/* TODO: needs a redesign? */
|
|
|
|
dbg->maps = r_debug_map_list_new ();
|
|
|
|
dbg->maps_user = r_debug_map_list_new ();
|
|
|
|
r_debug_signal_init (dbg);
|
|
|
|
if (hard) {
|
|
|
|
dbg->bp = r_bp_new ();
|
|
|
|
r_debug_plugin_init (dbg);
|
2015-09-14 02:08:31 +02:00
|
|
|
dbg->bp->iob.init = false;
|
2009-09-15 13:24:28 +02:00
|
|
|
}
|
2010-01-19 11:25:17 +01:00
|
|
|
return dbg;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2015-06-09 00:47:13 +02:00
|
|
|
static int free_tracenodes_entry (RDebug *dbg, const char *k, const char *v) {
|
|
|
|
ut64 v_num = r_num_get (NULL, v);
|
|
|
|
free((void *)(size_t)v_num);
|
2015-09-14 02:08:31 +02:00
|
|
|
return true;
|
2015-06-09 00:47:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API void r_debug_tracenodes_reset (RDebug *dbg) {
|
|
|
|
sdb_foreach (dbg->tracenodes, (SdbForeachCallback)free_tracenodes_entry, dbg);
|
|
|
|
sdb_reset (dbg->tracenodes);
|
|
|
|
}
|
|
|
|
|
2014-06-05 03:46:17 +02:00
|
|
|
R_API RDebug *r_debug_free(RDebug *dbg) {
|
2011-12-16 16:33:06 +01:00
|
|
|
if (!dbg) return NULL;
|
2010-09-20 14:02:45 +02:00
|
|
|
// TODO: free it correctly.. we must ensure this is an instance and not a reference..
|
2014-09-17 15:47:05 +02:00
|
|
|
r_bp_free (dbg->bp);
|
2010-01-21 02:38:52 +01:00
|
|
|
//r_reg_free(&dbg->reg);
|
2015-02-23 03:29:40 +01:00
|
|
|
r_list_free (dbg->snaps);
|
2015-08-25 21:29:08 +02:00
|
|
|
r_list_free (dbg->maps);
|
2015-08-29 04:58:57 -03:00
|
|
|
r_list_free (dbg->maps_user);
|
2015-08-29 14:01:38 +02:00
|
|
|
r_list_free (dbg->threads);
|
2014-04-23 03:54:06 +02:00
|
|
|
sdb_free (dbg->sgnls);
|
2015-06-09 00:47:13 +02:00
|
|
|
r_tree_free (dbg->tree);
|
|
|
|
sdb_foreach (dbg->tracenodes, (SdbForeachCallback)free_tracenodes_entry, dbg);
|
|
|
|
sdb_free (dbg->tracenodes);
|
2010-05-26 18:25:35 +02:00
|
|
|
//r_debug_plugin_free();
|
2015-08-29 04:58:57 -03:00
|
|
|
free (dbg->btalgo);
|
2013-11-17 14:25:45 +04:00
|
|
|
r_debug_trace_free (dbg);
|
2015-10-22 03:11:03 +02:00
|
|
|
free (dbg->arch);
|
2010-01-21 02:38:52 +01:00
|
|
|
free (dbg);
|
2009-08-22 03:11:33 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-05-06 00:59:10 +02:00
|
|
|
R_API int r_debug_attach(RDebug *dbg, int pid) {
|
2015-09-14 02:08:31 +02:00
|
|
|
int ret = false;
|
2010-01-21 02:38:52 +01:00
|
|
|
if (dbg && dbg->h && dbg->h->attach) {
|
2011-05-06 00:59:10 +02:00
|
|
|
ret = dbg->h->attach (dbg, pid);
|
2010-11-17 02:31:56 +01:00
|
|
|
if (ret != -1) {
|
2015-08-09 14:48:51 +02:00
|
|
|
eprintf ("Attached debugger to pid = %d, tid = %d\n", pid, ret);
|
2010-11-17 02:31:56 +01:00
|
|
|
r_debug_select (dbg, pid, ret); //dbg->pid, dbg->tid);
|
2016-02-06 19:53:29 +01:00
|
|
|
}
|
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-10-20 18:04:26 +02:00
|
|
|
/* stop execution of child process */
|
|
|
|
R_API int r_debug_stop(RDebug *dbg) {
|
|
|
|
if (dbg && dbg->h && dbg->h->stop)
|
|
|
|
return dbg->h->stop (dbg);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2011-10-20 18:04:26 +02:00
|
|
|
}
|
|
|
|
|
2015-12-04 02:57:56 +01:00
|
|
|
R_API bool r_debug_set_arch(RDebug *dbg, const char *arch, int bits) {
|
2015-10-22 03:11:03 +02:00
|
|
|
if (arch && dbg && dbg->h) {
|
2015-10-22 04:48:56 +02:00
|
|
|
bool rc = r_sys_arch_match (dbg->h->arch, arch);
|
2015-10-22 03:11:03 +02:00
|
|
|
if (rc) {
|
2011-09-22 09:52:00 +02:00
|
|
|
switch (bits) {
|
|
|
|
case 32:
|
2015-08-31 13:54:24 +02:00
|
|
|
if (dbg->h->bits & R_SYS_BITS_32) {
|
|
|
|
dbg->bits = R_SYS_BITS_32;
|
|
|
|
}
|
2011-09-22 09:52:00 +02:00
|
|
|
break;
|
|
|
|
case 64:
|
|
|
|
dbg->bits = R_SYS_BITS_64;
|
|
|
|
break;
|
|
|
|
}
|
2015-08-31 13:54:24 +02:00
|
|
|
if (!dbg->h->bits) {
|
2012-07-05 16:02:12 +00:00
|
|
|
dbg->bits = dbg->h->bits;
|
2015-08-31 13:54:24 +02:00
|
|
|
} else if (!(dbg->h->bits & dbg->bits)) {
|
2015-09-22 01:49:14 +02:00
|
|
|
dbg->bits = dbg->h->bits & R_SYS_BITS_64;
|
|
|
|
if (!dbg->bits)
|
|
|
|
dbg->bits = dbg->h->bits & R_SYS_BITS_32;
|
|
|
|
if (!dbg->bits)
|
|
|
|
dbg->bits = R_SYS_BITS_32;
|
|
|
|
eprintf ("Invalid value for bits\n");
|
2015-08-31 13:54:24 +02:00
|
|
|
}
|
2015-10-22 03:11:03 +02:00
|
|
|
free (dbg->arch);
|
|
|
|
dbg->arch = strdup (arch);
|
2015-09-14 02:08:31 +02:00
|
|
|
return true;
|
2011-05-06 00:59:10 +02:00
|
|
|
}
|
|
|
|
}
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2011-05-06 00:59:10 +02:00
|
|
|
}
|
|
|
|
|
2015-08-29 14:01:38 +02:00
|
|
|
/*
|
2010-02-05 12:21:37 +01:00
|
|
|
* Save 4096 bytes from %esp
|
|
|
|
* TODO: Add support for reverse stack architectures
|
2013-04-16 04:01:39 +02:00
|
|
|
* Also known as r_debug_inject()
|
2010-02-05 12:21:37 +01:00
|
|
|
*/
|
2013-04-16 04:01:39 +02:00
|
|
|
R_API ut64 r_debug_execute(RDebug *dbg, const ut8 *buf, int len, int restore) {
|
2010-02-05 12:21:37 +01:00
|
|
|
int orig_sz;
|
|
|
|
ut8 stackbackup[4096];
|
|
|
|
ut8 *backup, *orig = NULL;
|
2010-09-18 02:51:17 +02:00
|
|
|
RRegItem *ri, *risp, *ripc;
|
2010-02-05 12:21:37 +01:00
|
|
|
ut64 rsp, rpc, ra0 = 0LL;
|
2011-05-20 20:42:25 +02:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2010-02-05 12:21:37 +01:00
|
|
|
ripc = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_PC], R_REG_TYPE_GPR);
|
2013-04-16 04:01:39 +02:00
|
|
|
risp = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_SP], R_REG_TYPE_GPR);
|
2010-02-05 12:21:37 +01:00
|
|
|
if (ripc) {
|
2015-09-14 02:08:31 +02:00
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, false);
|
2010-02-05 12:21:37 +01:00
|
|
|
orig = r_reg_get_bytes (dbg->reg, -1, &orig_sz);
|
|
|
|
if (orig == NULL) {
|
|
|
|
eprintf ("Cannot get register arena bytes\n");
|
|
|
|
return 0LL;
|
|
|
|
}
|
|
|
|
rpc = r_reg_get_value (dbg->reg, ripc);
|
|
|
|
rsp = r_reg_get_value (dbg->reg, risp);
|
|
|
|
|
|
|
|
backup = malloc (len);
|
2014-04-25 23:04:02 +00:00
|
|
|
if (backup == NULL) {
|
|
|
|
free (orig);
|
2010-02-05 12:21:37 +01:00
|
|
|
return 0LL;
|
2014-04-25 23:04:02 +00:00
|
|
|
}
|
2010-02-05 12:21:37 +01:00
|
|
|
dbg->iob.read_at (dbg->iob.io, rpc, backup, len);
|
|
|
|
dbg->iob.read_at (dbg->iob.io, rsp, stackbackup, len);
|
|
|
|
|
2014-09-26 13:57:03 +02:00
|
|
|
r_bp_add_sw (dbg->bp, rpc+len, dbg->bpsize, R_BP_PROT_EXEC);
|
2010-02-05 12:21:37 +01:00
|
|
|
|
|
|
|
/* execute code here */
|
|
|
|
dbg->iob.write_at (dbg->iob.io, rpc, buf, len);
|
2013-04-16 04:01:39 +02:00
|
|
|
//r_bp_add_sw (dbg->bp, rpc+len, 4, R_BP_PROT_EXEC);
|
2010-02-05 12:21:37 +01:00
|
|
|
r_debug_continue (dbg);
|
2013-04-16 04:01:39 +02:00
|
|
|
//r_bp_del (dbg->bp, rpc+len);
|
2010-02-05 12:21:37 +01:00
|
|
|
/* TODO: check if stopped in breakpoint or not */
|
|
|
|
|
|
|
|
r_bp_del (dbg->bp, rpc+len);
|
|
|
|
dbg->iob.write_at (dbg->iob.io, rpc, backup, len);
|
2013-04-16 04:01:39 +02:00
|
|
|
if (restore) {
|
|
|
|
dbg->iob.write_at (dbg->iob.io, rsp, stackbackup, len);
|
|
|
|
}
|
2010-02-05 12:21:37 +01:00
|
|
|
|
2015-09-14 02:08:31 +02:00
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, false);
|
2010-02-05 12:21:37 +01:00
|
|
|
ri = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_A0], R_REG_TYPE_GPR);
|
|
|
|
ra0 = r_reg_get_value (dbg->reg, ri);
|
2013-04-16 04:01:39 +02:00
|
|
|
if (restore) {
|
|
|
|
r_reg_set_bytes (dbg->reg, -1, orig, orig_sz);
|
|
|
|
} else {
|
|
|
|
r_reg_set_value (dbg->reg, ripc, rpc);
|
|
|
|
}
|
2015-09-14 02:08:31 +02:00
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, true);
|
2010-02-05 12:21:37 +01:00
|
|
|
free (backup);
|
|
|
|
free (orig);
|
2010-04-14 13:02:23 +02:00
|
|
|
eprintf ("ra0=0x%08"PFMT64x"\n", ra0);
|
2010-02-05 12:21:37 +01:00
|
|
|
} else eprintf ("r_debug_execute: Cannot get program counter\n");
|
|
|
|
return (ra0);
|
|
|
|
}
|
|
|
|
|
2010-02-22 12:42:43 +01:00
|
|
|
R_API int r_debug_startv(struct r_debug_t *dbg, int argc, char **argv) {
|
2010-02-05 12:21:37 +01:00
|
|
|
/* TODO : r_debug_startv unimplemented */
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2015-11-17 00:57:22 +01:00
|
|
|
R_API int r_debug_start(RDebug *dbg, const char *cmd) {
|
2010-02-22 12:42:43 +01:00
|
|
|
/* TODO: this argc/argv parser is done in r_io */
|
2009-08-22 01:54:24 +00:00
|
|
|
// TODO: parse cmd and generate argc and argv
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2015-11-17 00:57:22 +01:00
|
|
|
R_API int r_debug_detach(RDebug *dbg, int pid) {
|
2009-02-05 22:08:46 +01:00
|
|
|
if (dbg->h && dbg->h->detach)
|
2016-01-22 10:53:58 +01:00
|
|
|
return dbg->h->detach (dbg, pid);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2010-04-11 01:46:07 +02:00
|
|
|
R_API int r_debug_select(RDebug *dbg, int pid, int tid) {
|
2014-08-31 18:09:25 +02:00
|
|
|
if (tid < 0)
|
|
|
|
tid = pid;
|
|
|
|
|
2015-08-17 00:12:54 +02:00
|
|
|
if (pid != -1 && tid != -1) {
|
|
|
|
if (pid != dbg->pid || tid != dbg->tid)
|
|
|
|
eprintf ("Debugging pid = %d, tid = %d now\n", pid, tid);
|
|
|
|
} else {
|
|
|
|
if (dbg->pid != -1)
|
|
|
|
eprintf ("Child %d is dead\n", dbg->pid);
|
|
|
|
}
|
2014-08-31 18:09:25 +02:00
|
|
|
|
|
|
|
if (dbg->h && dbg->h->select && !dbg->h->select (pid, tid))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-08-31 18:09:25 +02:00
|
|
|
|
2009-02-05 22:08:46 +01:00
|
|
|
dbg->pid = pid;
|
|
|
|
dbg->tid = tid;
|
2014-08-31 18:09:25 +02:00
|
|
|
|
2015-09-14 02:08:31 +02:00
|
|
|
return true;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2015-08-17 00:12:54 +02:00
|
|
|
R_API const char *r_debug_reason_to_string(int type) {
|
|
|
|
switch (type) {
|
|
|
|
case R_DEBUG_REASON_DEAD: return "dead";
|
2015-08-17 00:55:07 +02:00
|
|
|
case R_DEBUG_REASON_ABORT: return "abort";
|
|
|
|
case R_DEBUG_REASON_SEGFAULT: return "segfault";
|
2015-08-17 00:12:54 +02:00
|
|
|
case R_DEBUG_REASON_NONE: return "none";
|
|
|
|
case R_DEBUG_REASON_SIGNAL: return "signal";
|
|
|
|
case R_DEBUG_REASON_BREAKPOINT: return "breakpoint";
|
|
|
|
case R_DEBUG_REASON_READERR: return "read-error";
|
|
|
|
case R_DEBUG_REASON_WRITERR: return "write-error";
|
|
|
|
case R_DEBUG_REASON_DIVBYZERO: return "div-by-zero";
|
|
|
|
case R_DEBUG_REASON_ILLEGAL: return "illegal";
|
|
|
|
case R_DEBUG_REASON_UNKNOWN: return "unknown";
|
|
|
|
case R_DEBUG_REASON_ERROR: return "error";
|
|
|
|
case R_DEBUG_REASON_NEW_PID: return "new-pid";
|
|
|
|
case R_DEBUG_REASON_NEW_TID: return "new-tid";
|
|
|
|
case R_DEBUG_REASON_NEW_LIB: return "new-lib";
|
|
|
|
case R_DEBUG_REASON_EXIT_PID: return "exit-pid";
|
|
|
|
case R_DEBUG_REASON_EXIT_TID: return "exit-tid";
|
|
|
|
case R_DEBUG_REASON_EXIT_LIB: return "exit-lib";
|
|
|
|
case R_DEBUG_REASON_TRAP: return "trap";
|
|
|
|
case R_DEBUG_REASON_SWI: return "software-interrupt";
|
|
|
|
case R_DEBUG_REASON_INT: return "interrupt";
|
|
|
|
case R_DEBUG_REASON_FPU: return "fpu";
|
|
|
|
case R_DEBUG_REASON_STEP: return "step";
|
|
|
|
}
|
|
|
|
return "unhandled";
|
|
|
|
}
|
|
|
|
|
2010-07-16 00:02:33 +02:00
|
|
|
R_API int r_debug_stop_reason(RDebug *dbg) {
|
2009-02-05 22:08:46 +01:00
|
|
|
// TODO: return reason to stop debugging
|
|
|
|
// - new process
|
|
|
|
// - trap instruction
|
|
|
|
// - illegal instruction
|
|
|
|
// - fpu exception
|
2010-09-20 14:02:45 +02:00
|
|
|
// return dbg->reason
|
2015-08-17 00:12:54 +02:00
|
|
|
return dbg->reason.type;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2016-02-06 19:53:29 +01:00
|
|
|
/* Returns R_DEBUG_REASON_* */
|
2010-03-12 18:46:11 +01:00
|
|
|
R_API int r_debug_wait(RDebug *dbg) {
|
|
|
|
int ret = 0;
|
2014-05-10 02:47:50 +02:00
|
|
|
if (!dbg)
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.type = R_DEBUG_REASON_UNKNOWN;
|
2016-02-06 19:53:29 +01:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-08-17 00:12:54 +02:00
|
|
|
return dbg->reason.type = R_DEBUG_REASON_DEAD;
|
2014-05-10 02:47:50 +02:00
|
|
|
if (dbg->h && dbg->h->wait) {
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.type = R_DEBUG_REASON_UNKNOWN;
|
2011-07-06 11:48:16 +02:00
|
|
|
ret = dbg->h->wait (dbg, dbg->pid);
|
2009-04-15 11:09:36 +00:00
|
|
|
dbg->newstate = 1;
|
2011-05-20 20:42:25 +02:00
|
|
|
if (ret == -1) {
|
|
|
|
eprintf ("\n==> Process finished\n\n");
|
2013-09-18 02:11:23 +02:00
|
|
|
r_debug_select (dbg, -1, -1);
|
2011-05-20 20:42:25 +02:00
|
|
|
}
|
2010-06-04 00:56:44 +02:00
|
|
|
if (dbg->trace->enabled)
|
2010-03-12 18:46:11 +01:00
|
|
|
r_debug_trace_pc (dbg);
|
2015-09-28 03:20:53 +02:00
|
|
|
dbg->reason.type = ret;
|
2015-08-17 00:12:54 +02:00
|
|
|
if (ret == R_DEBUG_REASON_SIGNAL && dbg->reason.signum != -1) {
|
2013-09-18 02:11:23 +02:00
|
|
|
/* handle signal on continuations here */
|
2015-08-17 00:12:54 +02:00
|
|
|
int what = r_debug_signal_what (dbg, dbg->reason.signum);
|
|
|
|
const char *name = r_debug_signal_resolve_i (dbg, dbg->reason.signum);
|
2014-11-08 00:28:20 +01:00
|
|
|
if (name && strcmp ("SIGTRAP", name))
|
2015-01-12 02:52:12 +01:00
|
|
|
r_cons_printf ("[+] signal %d aka %s received %d\n",
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.signum, name, what);
|
2013-09-18 02:11:23 +02:00
|
|
|
}
|
2009-04-15 11:09:36 +00:00
|
|
|
}
|
2009-04-01 00:28:13 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-11-11 03:13:44 +01:00
|
|
|
R_API int r_debug_step_soft(RDebug *dbg) {
|
|
|
|
ut8 buf[32];
|
2014-09-15 14:38:28 +02:00
|
|
|
ut64 pc, sp;
|
|
|
|
ut64 next[2];
|
2010-11-11 03:13:44 +01:00
|
|
|
RAnalOp op;
|
2014-09-17 13:28:07 +02:00
|
|
|
int br, i, ret;
|
2014-09-15 14:38:28 +02:00
|
|
|
union {
|
|
|
|
ut64 r64;
|
|
|
|
ut32 r32[2];
|
2014-09-17 13:28:07 +02:00
|
|
|
} sp_top;
|
2014-09-15 14:38:28 +02:00
|
|
|
|
2011-05-20 20:42:25 +02:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-15 14:38:28 +02:00
|
|
|
|
|
|
|
pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]);
|
|
|
|
sp = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_SP]);
|
|
|
|
|
2015-07-21 06:06:00 +02:00
|
|
|
if (dbg->iob.read_at) {
|
|
|
|
if (dbg->iob.read_at (dbg->iob.io, pc, buf, sizeof (buf)) < 0)
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2016-02-06 19:53:29 +01:00
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
2014-09-15 14:38:28 +02:00
|
|
|
|
|
|
|
if (!r_anal_op (dbg->anal, &op, pc, buf, sizeof (buf)))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-15 14:38:28 +02:00
|
|
|
|
|
|
|
if (op.type == R_ANAL_OP_TYPE_ILL)
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-15 14:38:28 +02:00
|
|
|
|
|
|
|
switch (op.type) {
|
2015-07-21 06:06:00 +02:00
|
|
|
case R_ANAL_OP_TYPE_RET:
|
|
|
|
dbg->iob.read_at (dbg->iob.io, sp, (ut8 *)&sp_top, 8);
|
|
|
|
next[0] = (dbg->bits == R_SYS_BITS_32) ? sp_top.r32[0] : sp_top.r64;
|
|
|
|
br = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_ANAL_OP_TYPE_CJMP:
|
|
|
|
case R_ANAL_OP_TYPE_CCALL:
|
|
|
|
next[0] = op.jump;
|
|
|
|
next[1] = op.fail;
|
|
|
|
br = 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_ANAL_OP_TYPE_CALL:
|
|
|
|
case R_ANAL_OP_TYPE_JMP:
|
|
|
|
next[0] = op.jump;
|
|
|
|
br = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
next[0] = op.addr + op.size;
|
|
|
|
br = 1;
|
|
|
|
break;
|
2014-09-15 14:38:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < br; i++)
|
2014-09-26 13:57:03 +02:00
|
|
|
r_bp_add_sw (dbg->bp, next[i], dbg->bpsize, R_BP_PROT_EXEC);
|
2014-09-15 14:38:28 +02:00
|
|
|
|
2014-09-17 13:28:07 +02:00
|
|
|
ret = r_debug_continue (dbg);
|
2010-11-11 03:13:44 +01:00
|
|
|
|
2014-09-15 14:38:28 +02:00
|
|
|
for (i = 0; i < br; i++)
|
|
|
|
r_bp_del (dbg->bp, next[i]);
|
|
|
|
|
2014-09-17 13:28:07 +02:00
|
|
|
return ret;
|
2010-11-11 03:13:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_debug_step_hard(RDebug *dbg) {
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.type = R_DEBUG_REASON_STEP;
|
2011-05-20 20:42:25 +02:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2010-11-17 02:31:56 +01:00
|
|
|
if (!dbg->h->step (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-17 13:28:07 +02:00
|
|
|
return r_debug_wait (dbg);
|
2010-11-11 03:13:44 +01:00
|
|
|
}
|
|
|
|
|
2010-09-23 20:42:35 +02:00
|
|
|
R_API int r_debug_step(RDebug *dbg, int steps) {
|
2014-09-17 13:28:07 +02:00
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
if (!dbg || !dbg->h)
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.type = R_DEBUG_REASON_STEP;
|
2014-09-17 13:28:07 +02:00
|
|
|
|
2015-08-17 00:12:54 +02:00
|
|
|
if (r_debug_is_dead (dbg)) {
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2015-08-17 00:12:54 +02:00
|
|
|
}
|
2014-09-17 13:28:07 +02:00
|
|
|
|
2014-09-17 14:57:33 +02:00
|
|
|
if (steps < 1)
|
|
|
|
steps = 1;
|
|
|
|
|
2014-09-17 13:28:07 +02:00
|
|
|
for (i = 0; i < steps; i++) {
|
2016-02-06 19:53:29 +01:00
|
|
|
ret = dbg->swstep ?
|
2014-09-17 13:28:07 +02:00
|
|
|
r_debug_step_soft (dbg):
|
|
|
|
r_debug_step_hard (dbg);
|
|
|
|
if (!ret) {
|
|
|
|
eprintf ("Stepping failed!\n");
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2015-08-17 00:12:54 +02:00
|
|
|
} else {
|
|
|
|
dbg->steps++;
|
|
|
|
dbg->reason.type = R_DEBUG_REASON_STEP;
|
2015-08-29 14:01:38 +02:00
|
|
|
//dbg->reason.addr =
|
2015-08-17 00:12:54 +02:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-09-17 13:28:07 +02:00
|
|
|
|
|
|
|
return i;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2010-02-05 12:21:37 +01:00
|
|
|
R_API void r_debug_io_bind(RDebug *dbg, RIO *io) {
|
|
|
|
r_io_bind (io, &dbg->bp->iob);
|
|
|
|
r_io_bind (io, &dbg->iob);
|
|
|
|
}
|
|
|
|
|
2010-05-24 12:07:54 +02:00
|
|
|
R_API int r_debug_step_over(RDebug *dbg, int steps) {
|
2010-05-24 17:51:51 +02:00
|
|
|
RAnalOp op;
|
2014-09-17 14:57:33 +02:00
|
|
|
ut64 buf_pc, pc;
|
|
|
|
ut8 buf[DBG_BUF_SIZE];
|
|
|
|
int i;
|
|
|
|
|
2011-05-20 20:42:25 +02:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-17 14:57:33 +02:00
|
|
|
|
|
|
|
if (steps < 1)
|
|
|
|
steps = 1;
|
|
|
|
|
2011-10-09 04:15:32 +02:00
|
|
|
if (dbg->h && dbg->h->step_over) {
|
2014-09-17 14:57:33 +02:00
|
|
|
for (i = 0; i < steps; i++)
|
2011-10-09 04:15:32 +02:00
|
|
|
if (!dbg->h->step_over (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-17 14:57:33 +02:00
|
|
|
return i;
|
2011-10-09 04:15:32 +02:00
|
|
|
}
|
2014-09-17 14:57:33 +02:00
|
|
|
|
|
|
|
if (!dbg->anal || !dbg->reg)
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-17 14:57:33 +02:00
|
|
|
|
|
|
|
// Initial refill
|
|
|
|
buf_pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]);
|
|
|
|
dbg->iob.read_at (dbg->iob.io, buf_pc, buf, sizeof (buf));
|
|
|
|
|
|
|
|
for (i = 0; i < steps; i++) {
|
|
|
|
pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]);
|
2015-08-29 14:01:38 +02:00
|
|
|
// Try to keep the buffer full
|
|
|
|
if (pc - buf_pc > sizeof (buf)) {
|
2014-09-17 14:57:33 +02:00
|
|
|
buf_pc = pc;
|
|
|
|
dbg->iob.read_at (dbg->iob.io, buf_pc, buf, sizeof (buf));
|
2012-05-30 11:14:41 +02:00
|
|
|
}
|
2014-09-17 14:57:33 +02:00
|
|
|
// Analyze the opcode
|
|
|
|
if (!r_anal_op (dbg->anal, &op, pc, buf + (pc - buf_pc), sizeof (buf) - (pc - buf_pc))) {
|
|
|
|
eprintf ("Decode error at %"PFMT64x"\n", pc);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-17 14:57:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Skip over all the subroutine calls
|
|
|
|
if (op.type == R_ANAL_OP_TYPE_CALL ||
|
|
|
|
op.type == R_ANAL_OP_TYPE_CCALL ||
|
|
|
|
op.type == R_ANAL_OP_TYPE_UCALL ||
|
|
|
|
op.type == R_ANAL_OP_TYPE_UCCALL) {
|
|
|
|
|
|
|
|
// Use op.fail here instead of pc+op.size to enforce anal backends to fill in this field
|
|
|
|
if (!r_debug_continue_until (dbg, op.fail)) {
|
2014-09-17 15:18:54 +02:00
|
|
|
eprintf ("Could not step over call @ 0x%"PFMT64x"\n", pc);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-17 14:57:33 +02:00
|
|
|
}
|
2014-10-23 00:46:19 +02:00
|
|
|
} else if ((op.prefix & (R_ANAL_OP_PREFIX_REP | R_ANAL_OP_PREFIX_REPNE | R_ANAL_OP_PREFIX_LOCK))) {
|
|
|
|
//eprintf ("REP: skip to next instruction...\n");
|
|
|
|
if (!r_debug_continue_until (dbg, pc+op.size)) {
|
|
|
|
eprintf ("step over failed over rep\n");
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-10-23 00:46:19 +02:00
|
|
|
}
|
2014-09-17 14:57:33 +02:00
|
|
|
} else r_debug_step (dbg, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2015-08-13 14:02:11 +02:00
|
|
|
|
2015-05-05 18:08:01 +02:00
|
|
|
#if __WINDOWS__
|
2015-08-13 14:02:11 +02:00
|
|
|
void w32_break_process (void *);
|
2015-05-05 18:08:01 +02:00
|
|
|
#endif
|
2010-05-24 12:07:54 +02:00
|
|
|
R_API int r_debug_continue_kill(RDebug *dbg, int sig) {
|
2014-10-30 00:21:25 +01:00
|
|
|
ut64 pc;
|
2015-09-14 02:08:31 +02:00
|
|
|
int retwait, ret = false;
|
2014-05-10 02:47:20 +02:00
|
|
|
if (!dbg)
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2015-05-05 18:08:01 +02:00
|
|
|
#if __WINDOWS__
|
2015-08-13 14:02:11 +02:00
|
|
|
r_cons_break(w32_break_process, dbg);
|
2015-05-05 18:08:01 +02:00
|
|
|
#endif
|
2014-10-30 00:21:25 +01:00
|
|
|
repeat:
|
2011-05-20 20:42:25 +02:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-05-10 02:47:20 +02:00
|
|
|
if (dbg->h && dbg->h->cont) {
|
2015-09-14 02:08:31 +02:00
|
|
|
r_bp_restore (dbg->bp, true); // set sw breakpoints
|
2011-04-03 16:38:24 +02:00
|
|
|
ret = dbg->h->cont (dbg, dbg->pid, dbg->tid, sig);
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.signum = 0;
|
2014-12-12 00:34:53 +01:00
|
|
|
retwait = r_debug_wait (dbg);
|
2015-05-05 18:08:01 +02:00
|
|
|
#if __WINDOWS__
|
2015-08-17 00:12:54 +02:00
|
|
|
if (retwait != R_DEBUG_REASON_DEAD) {
|
2015-08-12 12:57:36 +02:00
|
|
|
ret = dbg->tid;
|
|
|
|
}
|
2016-01-22 10:53:58 +01:00
|
|
|
if (retwait == R_DEBUG_REASON_NEW_LIB ||
|
|
|
|
retwait == R_DEBUG_REASON_EXIT_LIB) {
|
2015-10-15 22:50:06 +02:00
|
|
|
goto repeat;
|
|
|
|
}
|
2015-05-05 18:08:01 +02:00
|
|
|
#endif
|
2016-02-06 19:53:29 +01:00
|
|
|
if (r_debug_is_dead (dbg))
|
|
|
|
return false;
|
2015-09-14 02:08:31 +02:00
|
|
|
r_bp_restore (dbg->bp, false); // unset sw breakpoints
|
2015-08-17 00:12:54 +02:00
|
|
|
if (r_debug_recoil (dbg) || (dbg->reason.type == R_DEBUG_REASON_BREAKPOINT)) {
|
2014-10-30 00:21:25 +01:00
|
|
|
/* check if cur bp demands tracing or not */
|
2014-10-30 22:58:51 +01:00
|
|
|
pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]);
|
2014-10-30 00:21:25 +01:00
|
|
|
RBreakpointItem *b = r_bp_get_at (dbg->bp, pc);
|
|
|
|
if (b) {
|
|
|
|
/* check if cur bp demands tracing or not */
|
2016-02-06 19:53:29 +01:00
|
|
|
if (b->trace)
|
2014-10-30 00:21:25 +01:00
|
|
|
eprintf("hit tracepoit at: %"PFMT64x"\n",pc);
|
2016-02-06 19:53:29 +01:00
|
|
|
else
|
2014-10-30 22:58:51 +01:00
|
|
|
eprintf("hit breakpoint at: %"PFMT64x"\n",pc);
|
2015-01-30 19:07:33 +01:00
|
|
|
if (dbg->trace->enabled)
|
|
|
|
r_debug_trace_pc (dbg);
|
2014-10-30 22:58:51 +01:00
|
|
|
// TODO: delegate this to RCore.bphit(RCore, RBreakopintItem)
|
|
|
|
if (dbg->corebind.core && dbg->corebind.bphit) {
|
|
|
|
dbg->corebind.bphit (dbg->corebind.core, b);
|
|
|
|
}
|
|
|
|
if (b->trace) {
|
2014-10-30 00:21:25 +01:00
|
|
|
r_debug_step (dbg, 1);
|
|
|
|
goto repeat;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-03-11 01:04:59 +01:00
|
|
|
r_debug_select (dbg, dbg->pid, ret);
|
2014-12-12 00:34:53 +01:00
|
|
|
sig = 0; // clear continuation after signal if needed
|
2015-08-17 00:12:54 +02:00
|
|
|
if (retwait == R_DEBUG_REASON_SIGNAL && dbg->reason.signum != -1) {
|
|
|
|
int what = r_debug_signal_what (dbg, dbg->reason.signum);
|
2014-12-12 00:34:53 +01:00
|
|
|
if (what & R_DBG_SIGNAL_CONT) {
|
2015-08-17 00:12:54 +02:00
|
|
|
sig = dbg->reason.signum;
|
2014-12-12 00:34:53 +01:00
|
|
|
eprintf ("Continue into the signal %d handler\n", sig);
|
|
|
|
goto repeat;
|
|
|
|
} else if (what & R_DBG_SIGNAL_SKIP) {
|
|
|
|
// skip signal. requires skipping one instruction
|
|
|
|
ut8 buf[64];
|
|
|
|
RAnalOp op = {0};
|
|
|
|
ut64 pc = r_debug_reg_get (dbg, "pc");
|
|
|
|
dbg->iob.read_at (dbg->iob.io, pc, buf, sizeof (buf));
|
|
|
|
r_anal_op (dbg->anal, &op, pc, buf, sizeof (buf));
|
2016-02-06 19:53:29 +01:00
|
|
|
if (op.size > 0) {
|
2015-08-17 00:12:54 +02:00
|
|
|
const char *signame = r_debug_signal_resolve_i (dbg, dbg->reason.signum);
|
2014-12-12 00:34:53 +01:00
|
|
|
r_debug_reg_set (dbg, "pc", pc+op.size);
|
|
|
|
eprintf ("Skip signal %d handler %s\n",
|
2015-08-17 00:12:54 +02:00
|
|
|
dbg->reason.signum, signame);
|
2014-12-12 00:34:53 +01:00
|
|
|
goto repeat;
|
2016-02-06 19:53:29 +01:00
|
|
|
} else {
|
2014-12-12 00:34:53 +01:00
|
|
|
ut64 pc = r_debug_reg_get (dbg, "pc");
|
|
|
|
eprintf ("Stalled with an exception at 0x%08"PFMT64x"\n", pc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-03-22 02:36:00 +00:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-05-24 12:07:54 +02:00
|
|
|
R_API int r_debug_continue(RDebug *dbg) {
|
2015-08-17 00:12:54 +02:00
|
|
|
return r_debug_continue_kill (dbg, 0); //dbg->reason.signum);
|
2010-01-19 11:25:17 +01:00
|
|
|
}
|
|
|
|
|
2010-06-04 00:56:44 +02:00
|
|
|
R_API int r_debug_continue_until_nontraced(RDebug *dbg) {
|
|
|
|
eprintf ("TODO\n");
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2010-06-04 00:56:44 +02:00
|
|
|
}
|
|
|
|
|
2010-05-24 17:51:51 +02:00
|
|
|
R_API int r_debug_continue_until_optype(RDebug *dbg, int type, int over) {
|
|
|
|
int ret, n = 0;
|
2014-09-15 14:38:28 +02:00
|
|
|
ut64 pc, buf_pc = 0;
|
2010-05-24 12:07:54 +02:00
|
|
|
RAnalOp op;
|
2014-09-17 14:57:33 +02:00
|
|
|
ut8 buf[DBG_BUF_SIZE];
|
2012-05-30 11:14:41 +02:00
|
|
|
|
2015-08-17 00:12:54 +02:00
|
|
|
if (r_debug_is_dead (dbg)) {
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2015-08-17 00:12:54 +02:00
|
|
|
}
|
2014-09-15 14:38:28 +02:00
|
|
|
|
2015-08-29 14:01:38 +02:00
|
|
|
if (!dbg->anal || !dbg->reg) {
|
2014-09-15 14:38:28 +02:00
|
|
|
eprintf ("Undefined pointer at dbg->anal\n");
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-15 14:38:28 +02:00
|
|
|
}
|
|
|
|
|
2014-10-13 14:52:33 +02:00
|
|
|
r_debug_step (dbg, 1);
|
2015-09-14 02:08:31 +02:00
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, false);
|
2014-10-13 14:52:33 +02:00
|
|
|
|
2014-09-15 14:38:28 +02:00
|
|
|
// Initial refill
|
|
|
|
buf_pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]);
|
|
|
|
dbg->iob.read_at (dbg->iob.io, buf_pc, buf, sizeof (buf));
|
|
|
|
|
2014-10-13 05:02:24 +02:00
|
|
|
// step first, we dont want to check current optype
|
2014-09-15 14:38:28 +02:00
|
|
|
for (;;) {
|
2015-09-14 02:08:31 +02:00
|
|
|
r_debug_reg_sync (dbg, R_REG_TYPE_GPR, false);
|
2014-09-15 14:38:28 +02:00
|
|
|
pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]);
|
2015-08-29 14:01:38 +02:00
|
|
|
// Try to keep the buffer full
|
|
|
|
if (pc - buf_pc > sizeof (buf)) {
|
2014-09-15 14:38:28 +02:00
|
|
|
buf_pc = pc;
|
|
|
|
dbg->iob.read_at (dbg->iob.io, buf_pc, buf, sizeof (buf));
|
|
|
|
}
|
|
|
|
// Analyze the opcode
|
|
|
|
if (!r_anal_op (dbg->anal, &op, pc, buf + (pc - buf_pc), sizeof (buf) - (pc - buf_pc))) {
|
|
|
|
eprintf ("Decode error at %"PFMT64x"\n", pc);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-15 14:38:28 +02:00
|
|
|
}
|
2015-08-29 14:01:38 +02:00
|
|
|
if (op.type == type)
|
2014-09-15 14:38:28 +02:00
|
|
|
break;
|
|
|
|
// Step over and repeat
|
|
|
|
ret = over ?
|
|
|
|
r_debug_step_over (dbg, 1) :
|
|
|
|
r_debug_step (dbg, 1);
|
|
|
|
|
|
|
|
if (!ret) {
|
|
|
|
eprintf ("r_debug_step: failed\n");
|
|
|
|
break;
|
2012-05-30 11:14:41 +02:00
|
|
|
}
|
2014-09-15 14:38:28 +02:00
|
|
|
n++;
|
|
|
|
}
|
|
|
|
|
2010-05-24 12:07:54 +02:00
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2014-08-01 12:07:08 +02:00
|
|
|
R_API int r_debug_continue_until(RDebug *dbg, ut64 addr) {
|
2014-09-16 15:09:22 +02:00
|
|
|
int has_bp;
|
|
|
|
ut64 pc;
|
|
|
|
|
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-09-16 15:09:22 +02:00
|
|
|
|
|
|
|
// Check if there was another breakpoint set at addr
|
2014-10-28 02:28:58 +01:00
|
|
|
has_bp = r_bp_get_in (dbg->bp, addr, R_BP_PROT_EXEC) != NULL;
|
2014-09-16 15:09:22 +02:00
|
|
|
if (!has_bp)
|
2014-09-26 13:57:03 +02:00
|
|
|
r_bp_add_sw (dbg->bp, addr, dbg->bpsize, R_BP_PROT_EXEC);
|
2014-09-16 15:09:22 +02:00
|
|
|
|
|
|
|
// Continue until the bp is reached
|
|
|
|
for (;;) {
|
|
|
|
if (r_debug_is_dead (dbg))
|
|
|
|
break;
|
|
|
|
|
|
|
|
pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]);
|
|
|
|
if (pc == addr)
|
|
|
|
break;
|
2015-01-12 17:47:34 +01:00
|
|
|
if (r_bp_get_at (dbg->bp, pc))
|
|
|
|
break;
|
2014-09-16 15:09:22 +02:00
|
|
|
r_debug_continue (dbg);
|
2014-08-01 12:07:08 +02:00
|
|
|
}
|
2014-09-16 15:09:22 +02:00
|
|
|
|
|
|
|
// Clean up if needed
|
2014-09-17 16:57:41 +02:00
|
|
|
if (!has_bp)
|
2014-09-16 15:09:22 +02:00
|
|
|
r_bp_del (dbg->bp, addr);
|
|
|
|
|
2015-09-14 02:08:31 +02:00
|
|
|
return true;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
2015-04-18 10:35:57 +02:00
|
|
|
static int show_syscall(RDebug *dbg, const char *sysreg) {
|
2014-06-14 02:01:03 +02:00
|
|
|
const char *sysname;
|
2015-04-18 10:35:57 +02:00
|
|
|
char regname[8];
|
|
|
|
int reg, i, args;
|
|
|
|
RSyscallItem *si;
|
|
|
|
reg = (int)r_debug_reg_get (dbg, sysreg);
|
|
|
|
si = r_syscall_get (dbg->anal->syscall, reg, -1);
|
|
|
|
if (si) {
|
|
|
|
sysname = si->name? si->name: "unknown";
|
|
|
|
args = si->args;
|
|
|
|
} else {
|
|
|
|
sysname = "unknown";
|
|
|
|
args = 3;
|
|
|
|
}
|
|
|
|
eprintf ("--> %s 0x%08"PFMT64x" syscall %d %s (", sysreg,
|
|
|
|
r_debug_reg_get (dbg, "pc"), reg, sysname);
|
|
|
|
for (i=0; i<args; i++) {
|
|
|
|
ut64 val;
|
2015-10-31 01:57:52 +01:00
|
|
|
snprintf (regname, sizeof (regname)-1, "A%d", i);
|
2015-04-18 10:35:57 +02:00
|
|
|
val = r_debug_reg_get (dbg, regname);
|
|
|
|
if (((st64)val<0) && ((st64)val>-0xffff)) {
|
|
|
|
eprintf ("%"PFMT64d"%s", val, (i+1==args)?"":" ");
|
|
|
|
} else {
|
|
|
|
eprintf ("0x%"PFMT64x"%s", val, (i+1==args)?"":" ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
eprintf (")\n");
|
|
|
|
r_syscall_item_free (si);
|
|
|
|
return reg;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_debug_continue_syscalls(RDebug *dbg, int *sc, int n_sc) {
|
2015-09-14 02:08:31 +02:00
|
|
|
int i, reg, ret = false;
|
2014-06-14 02:01:03 +02:00
|
|
|
if (!dbg || !dbg->h || r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-06-14 02:01:03 +02:00
|
|
|
if (!dbg->h->contsc) {
|
|
|
|
/* user-level syscall tracing */
|
|
|
|
r_debug_continue_until_optype (dbg, R_ANAL_OP_TYPE_SWI, 0);
|
2015-10-31 01:57:52 +01:00
|
|
|
return show_syscall (dbg, "A0");
|
2014-06-14 02:01:03 +02:00
|
|
|
}
|
|
|
|
|
2015-09-14 02:08:31 +02:00
|
|
|
if (!r_debug_reg_sync (dbg, R_REG_TYPE_GPR, false)) {
|
2014-06-14 02:01:03 +02:00
|
|
|
eprintf ("--> cannot read registers\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2014-11-20 01:30:37 +01:00
|
|
|
{
|
|
|
|
int err;
|
2016-01-13 22:25:08 +01:00
|
|
|
reg = (int)r_debug_reg_get_err (dbg, "SN", &err);
|
2014-11-20 01:30:37 +01:00
|
|
|
if (err) {
|
|
|
|
eprintf ("Cannot find 'sn' register for current arch-os.\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2014-06-14 02:01:03 +02:00
|
|
|
}
|
|
|
|
for (;;) {
|
2015-04-19 13:39:10 +02:00
|
|
|
if (r_cons_singleton()->breaked)
|
|
|
|
break;
|
|
|
|
#if __linux__
|
|
|
|
// step is needed to avoid dupped contsc results
|
|
|
|
r_debug_step (dbg, 1);
|
|
|
|
#endif
|
2014-06-14 02:01:03 +02:00
|
|
|
dbg->h->contsc (dbg, dbg->pid, 0); // TODO handle return value
|
2014-10-09 18:31:55 +02:00
|
|
|
// wait until continuation
|
|
|
|
r_debug_wait (dbg);
|
2015-09-14 02:08:31 +02:00
|
|
|
if (!r_debug_reg_sync (dbg, R_REG_TYPE_GPR, false)) {
|
2015-04-19 13:39:10 +02:00
|
|
|
eprintf ("--> cannot sync regs, process is probably dead\n");
|
2014-06-14 02:01:03 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2016-01-13 22:25:08 +01:00
|
|
|
reg = show_syscall (dbg, "SN");
|
2014-10-09 18:31:55 +02:00
|
|
|
if (n_sc == -1)
|
|
|
|
continue;
|
|
|
|
if (n_sc == 0) {
|
|
|
|
break;
|
|
|
|
}
|
2014-06-14 02:27:43 +02:00
|
|
|
for (i=0; i<n_sc; i++) {
|
2014-06-14 02:01:03 +02:00
|
|
|
if (sc[i] == reg)
|
|
|
|
return reg;
|
2010-05-24 12:07:54 +02:00
|
|
|
}
|
2014-06-14 02:01:03 +02:00
|
|
|
// TODO: must use r_core_cmd(as)..import code from rcore
|
2010-03-04 01:46:25 +01:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-06-14 02:01:03 +02:00
|
|
|
R_API int r_debug_continue_syscall(RDebug *dbg, int sc) {
|
|
|
|
return r_debug_continue_syscalls (dbg, &sc, 1);
|
|
|
|
}
|
|
|
|
|
2010-01-21 02:38:52 +01:00
|
|
|
// TODO: remove from here? this is code injection!
|
2012-12-07 12:03:53 +01:00
|
|
|
R_API int r_debug_syscall(RDebug *dbg, int num) {
|
2015-12-01 12:39:12 +01:00
|
|
|
bool ret = false;
|
2010-05-24 12:07:54 +02:00
|
|
|
if (dbg->h->contsc) {
|
2011-04-03 16:38:24 +02:00
|
|
|
ret = dbg->h->contsc (dbg, dbg->pid, num);
|
2010-05-24 12:07:54 +02:00
|
|
|
} else {
|
2015-09-14 02:08:31 +02:00
|
|
|
ret = true;
|
2010-05-24 12:07:54 +02:00
|
|
|
// TODO.check for num
|
|
|
|
}
|
2010-09-23 20:42:35 +02:00
|
|
|
eprintf ("TODO: show syscall information\n");
|
2010-03-12 18:46:11 +01:00
|
|
|
/* r2rc task? ala inject? */
|
2015-09-14 02:08:31 +02:00
|
|
|
return (int)ret;
|
2009-09-10 20:51:34 +00:00
|
|
|
}
|
|
|
|
|
2012-12-07 12:03:53 +01:00
|
|
|
R_API int r_debug_kill(RDebug *dbg, int pid, int tid, int sig) {
|
2015-09-14 02:08:31 +02:00
|
|
|
int ret = false;
|
2013-03-01 01:57:36 +02:00
|
|
|
if (r_debug_is_dead (dbg))
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2010-02-28 22:58:21 +01:00
|
|
|
if (dbg->h && dbg->h->kill)
|
2012-12-07 12:03:53 +01:00
|
|
|
ret = dbg->h->kill (dbg, pid, tid, sig);
|
2010-02-28 22:58:21 +01:00
|
|
|
else eprintf ("Backend does not implements kill()\n");
|
|
|
|
return ret;
|
2009-04-01 22:44:43 +00:00
|
|
|
}
|
2010-02-02 11:09:52 +01:00
|
|
|
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API RList *r_debug_frames(RDebug *dbg, ut64 at) {
|
2010-03-02 11:18:49 +01:00
|
|
|
if (dbg && dbg->h && dbg->h->frames)
|
2012-06-14 02:18:15 +02:00
|
|
|
return dbg->h->frames (dbg, at);
|
2010-03-02 11:18:49 +01:00
|
|
|
return NULL;
|
2010-02-02 11:09:52 +01:00
|
|
|
}
|
2010-11-18 11:41:17 +01:00
|
|
|
|
|
|
|
/* TODO: Implement fork and clone */
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API int r_debug_child_fork(RDebug *dbg) {
|
2010-11-18 11:41:17 +01:00
|
|
|
//if (dbg && dbg->h && dbg->h->frames)
|
|
|
|
//return dbg->h->frames (dbg);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API int r_debug_child_clone(RDebug *dbg) {
|
2010-11-18 11:41:17 +01:00
|
|
|
//if (dbg && dbg->h && dbg->h->frames)
|
|
|
|
//return dbg->h->frames (dbg);
|
|
|
|
return 0;
|
|
|
|
}
|
2011-05-20 20:42:25 +02:00
|
|
|
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API int r_debug_is_dead(RDebug *dbg) {
|
2015-08-17 00:12:54 +02:00
|
|
|
int is_dead = (dbg->pid == -1);
|
|
|
|
if (is_dead) {
|
|
|
|
dbg->reason.type = R_DEBUG_REASON_DEAD;
|
|
|
|
}
|
|
|
|
return is_dead;
|
2011-05-20 20:42:25 +02:00
|
|
|
}
|
2011-11-25 04:32:32 +01:00
|
|
|
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API int r_debug_map_protect(RDebug *dbg, ut64 addr, int size, int perms) {
|
2011-11-25 04:32:32 +01:00
|
|
|
if (dbg && dbg->h && dbg->h->map_protect)
|
|
|
|
return dbg->h->map_protect (dbg, addr, size, perms);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2011-11-25 04:32:32 +01:00
|
|
|
}
|
2014-01-28 04:38:02 +01:00
|
|
|
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API void r_debug_drx_list(RDebug *dbg) {
|
2014-01-28 04:38:02 +01:00
|
|
|
if (dbg && dbg->h && dbg->h->drx)
|
|
|
|
dbg->h->drx (dbg, 0, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API int r_debug_drx_set(RDebug *dbg, int idx, ut64 addr, int len, int rwx, int g) {
|
2014-01-28 04:38:02 +01:00
|
|
|
if (dbg && dbg->h && dbg->h->drx)
|
|
|
|
return dbg->h->drx (dbg, idx, addr, len, rwx, g);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-01-28 04:38:02 +01:00
|
|
|
}
|
|
|
|
|
2016-01-29 18:52:53 -05:00
|
|
|
R_API int r_debug_drx_unset(RDebug *dbg, int idx) {
|
2014-01-28 04:38:02 +01:00
|
|
|
if (dbg && dbg->h && dbg->h->drx)
|
|
|
|
return dbg->h->drx (dbg, idx, 0, -1, 0, 0);
|
2015-09-14 02:08:31 +02:00
|
|
|
return false;
|
2014-01-28 04:38:02 +01:00
|
|
|
}
|