/* radare - LGPL - Copyright 2009-2014 - pancake, TheLemonMan */ #include #include #include R_LIB_VERSION(r_debug); // Size of the lookahead buffers used in r_debug functions #define DBG_BUF_SIZE 512 R_API RDebugInfo *r_debug_info(RDebug *dbg, const char *arg) { if (!dbg || !dbg->h || !dbg->h->info) 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); } /* restore program counter after breakpoint hit */ static int r_debug_recoil(RDebug *dbg) { int recoil; RRegItem *ri; if (r_debug_is_dead (dbg)) return R_FALSE; r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_FALSE); ri = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_PC], -1); if (ri) { ut64 addr = r_reg_get_value (dbg->reg, ri); recoil = r_bp_recoil (dbg->bp, addr); //eprintf ("[R2] Breakpoint recoil at 0x%"PFMT64x" = %d\n", addr, recoil); #if __arm__ if (recoil<1) recoil = 0; // XXX Hack :D #else if (recoil<1) recoil = 0; //1; // XXX Hack :D (x86 only?) #endif if (recoil) { dbg->reason = R_DBG_REASON_BP; r_reg_set_value (dbg->reg, ri, addr-recoil); if (r_reg_get_value (dbg->reg, ri ) != (addr-recoil)) { eprintf ("r_debug_recoil: Cannot set program counter\n"); return R_FALSE; } r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_TRUE); //eprintf ("[BP Hit] Setting pc to 0x%"PFMT64x"\n", (addr-recoil)); return R_TRUE; } } else eprintf ("r_debug_recoil: Cannot get program counter\n"); return R_FALSE; } R_API RDebug *r_debug_new(int hard) { RDebug *dbg = R_NEW (RDebug); if (dbg) { // R_SYS_ARCH dbg->arch = r_sys_arch_id (R_SYS_ARCH); // 0 is native by default dbg->bits = R_SYS_BITS; dbg->anal = NULL; dbg->pid = -1; dbg->tid = -1; dbg->graph = r_graph_new (); dbg->swstep = 0; dbg->newstate = 0; dbg->signum = 0; dbg->reason = R_DBG_REASON_UNKNOWN; dbg->stop_all_threads = R_FALSE; dbg->trace = r_debug_trace_new (); dbg->printf = (void *)printf; dbg->reg = r_reg_new (); dbg->h = NULL; /* 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); dbg->bp->iob.init = R_FALSE; } } return dbg; } R_API RDebug *r_debug_free(RDebug *dbg) { if (!dbg) return NULL; // TODO: free it correctly.. we must ensure this is an instance and not a reference.. //r_bp_free(&dbg->bp); //r_reg_free(&dbg->reg); sdb_free (dbg->sgnls); //r_debug_plugin_free(); r_debug_trace_free (dbg); r_graph_free (dbg->graph); free (dbg); return NULL; } R_API int r_debug_attach(RDebug *dbg, int pid) { int ret = R_FALSE; if (dbg && dbg->h && dbg->h->attach) { ret = dbg->h->attach (dbg, pid); if (ret != -1) { eprintf ("pid = %d tid = %d\n", pid, ret); // TODO: get arch and set io pid //int arch = dbg->h->arch; //r_reg_set(dbg->reg->nregs, arch); //R_DBG_ARCH_X86); // dbg->bp->iob->system("pid %d", pid); //dbg->pid = pid; //dbg->tid = ret; r_debug_select (dbg, pid, ret); //dbg->pid, dbg->tid); }// else if (pid != -1) // eprintf ("Cannot attach to this pid %d\n", pid); } else eprintf ("dbg->attach = NULL\n"); return ret; } /* 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); return R_FALSE; } R_API int r_debug_set_arch(RDebug *dbg, int arch, int bits) { if (dbg && dbg->h) { if (arch & dbg->h->arch) { //eprintf ("arch supported by debug backend (%x)\n", arch); switch (bits) { case 32: dbg->bits = R_SYS_BITS_32; break; case 64: dbg->bits = R_SYS_BITS_64; break; } if (!(dbg->h->bits & dbg->bits)) dbg->bits = dbg->h->bits; dbg->arch = arch; return R_TRUE; } //eprintf ("arch (%s, %d) not supported by debug backend\n", // r_sys_arch_str (arch), bits); } return R_FALSE; } /* * Save 4096 bytes from %esp * TODO: Add support for reverse stack architectures * Also known as r_debug_inject() */ R_API ut64 r_debug_execute(RDebug *dbg, const ut8 *buf, int len, int restore) { int orig_sz; ut8 stackbackup[4096]; ut8 *backup, *orig = NULL; RRegItem *ri, *risp, *ripc; ut64 rsp, rpc, ra0 = 0LL; if (r_debug_is_dead (dbg)) return R_FALSE; ripc = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_PC], R_REG_TYPE_GPR); risp = r_reg_get (dbg->reg, dbg->reg->name[R_REG_NAME_SP], R_REG_TYPE_GPR); if (ripc) { r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_FALSE); 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); if (backup == NULL) { free (orig); return 0LL; } dbg->iob.read_at (dbg->iob.io, rpc, backup, len); dbg->iob.read_at (dbg->iob.io, rsp, stackbackup, len); r_bp_add_sw (dbg->bp, rpc+len, 1, R_BP_PROT_EXEC); /* execute code here */ dbg->iob.write_at (dbg->iob.io, rpc, buf, len); //r_bp_add_sw (dbg->bp, rpc+len, 4, R_BP_PROT_EXEC); r_debug_continue (dbg); //r_bp_del (dbg->bp, rpc+len); /* 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); if (restore) { dbg->iob.write_at (dbg->iob.io, rsp, stackbackup, len); } r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_FALSE); 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); if (restore) { r_reg_set_bytes (dbg->reg, -1, orig, orig_sz); } else { r_reg_set_value (dbg->reg, ripc, rpc); } r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_TRUE); free (backup); free (orig); eprintf ("ra0=0x%08"PFMT64x"\n", ra0); } else eprintf ("r_debug_execute: Cannot get program counter\n"); return (ra0); } R_API int r_debug_startv(struct r_debug_t *dbg, int argc, char **argv) { /* TODO : r_debug_startv unimplemented */ return R_FALSE; } R_API int r_debug_start(struct r_debug_t *dbg, const char *cmd) { /* TODO: this argc/argv parser is done in r_io */ // TODO: parse cmd and generate argc and argv return R_FALSE; } R_API int r_debug_detach(struct r_debug_t *dbg, int pid) { if (dbg->h && dbg->h->detach) return dbg->h->detach(pid); return R_FALSE; } R_API int r_debug_select(RDebug *dbg, int pid, int tid) { if (!tid) tid = pid; if (pid != dbg->pid || tid != dbg->tid) eprintf ("r_debug_select: %d %d\n", pid, tid); dbg->pid = pid; if (tid == -1) tid = dbg->pid; dbg->tid = tid; return R_TRUE; } R_API int r_debug_stop_reason(RDebug *dbg) { // TODO: return reason to stop debugging // - new process // - trap instruction // - illegal instruction // - fpu exception // return dbg->reason return dbg->reason; } /* Returns PID */ R_API int r_debug_wait(RDebug *dbg) { int ret = 0; if (!dbg) return R_FALSE; if (r_debug_is_dead (dbg)) return R_FALSE; if (dbg->h && dbg->h->wait) { dbg->reason = R_DBG_REASON_UNKNOWN; ret = dbg->h->wait (dbg, dbg->pid); dbg->reason = ret; dbg->newstate = 1; if (ret == -1) { eprintf ("\n==> Process finished\n\n"); r_debug_select (dbg, -1, -1); } //eprintf ("wait = %d\n", ret); if (dbg->trace->enabled) r_debug_trace_pc (dbg); if (ret == R_DBG_REASON_SIGNAL && dbg->signum != -1) { /* handle signal on continuations here */ int what = r_debug_signal_what (dbg, dbg->signum); const char *name = r_debug_signal_resolve_i (dbg, dbg->signum); if (strcmp ("SIGTRAP", name)) r_cons_printf ("[+] signal %d aka %s received\n", dbg->signum, name); if (what & R_DBG_SIGNAL_SKIP) { dbg->signum = 0; // TODO: use ptrace-setsiginfo to ignore signal } if (what & R_DBG_SIGNAL_CONT) { // XXX: support step, steptrace, continue_until_foo, etc.. r_debug_continue (dbg); } } } return ret; } R_API int r_debug_step_soft(RDebug *dbg) { ut8 buf[32]; ut64 pc, sp; ut64 next[2]; RAnalOp op; int br, i, ret; union { ut64 r64; ut32 r32[2]; } sp_top; if (r_debug_is_dead (dbg)) return R_FALSE; 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]); if (dbg->iob.read_at (dbg->iob.io, pc, buf, sizeof (buf)) < 0) return R_FALSE; if (!r_anal_op (dbg->anal, &op, pc, buf, sizeof (buf))) return R_FALSE; if (op.type == R_ANAL_OP_TYPE_ILL) return R_FALSE; switch (op.type) { 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; } for (i = 0; i < br; i++) r_bp_add_sw (dbg->bp, next[i], 1, R_BP_PROT_EXEC); ret = r_debug_continue (dbg); for (i = 0; i < br; i++) r_bp_del (dbg->bp, next[i]); return ret; } R_API int r_debug_step_hard(RDebug *dbg) { if (r_debug_is_dead (dbg)) return R_FALSE; if (!dbg->h->step (dbg)) return R_FALSE; return r_debug_wait (dbg); } R_API int r_debug_step(RDebug *dbg, int steps) { int i, ret; if (!dbg || !dbg->h) return R_FALSE; if (r_debug_is_dead (dbg)) return R_FALSE; if (steps < 1) steps = 1; for (i = 0; i < steps; i++) { ret = dbg->swstep? r_debug_step_soft (dbg): r_debug_step_hard (dbg); if (!ret) { eprintf ("Stepping failed!\n"); return R_FALSE; } else dbg->steps++; } return i; } R_API void r_debug_io_bind(RDebug *dbg, RIO *io) { r_io_bind (io, &dbg->bp->iob); r_io_bind (io, &dbg->iob); } R_API int r_debug_step_over(RDebug *dbg, int steps) { RAnalOp op; ut64 buf_pc, pc; ut8 buf[DBG_BUF_SIZE]; int i; if (r_debug_is_dead (dbg)) return R_FALSE; if (steps < 1) steps = 1; if (dbg->h && dbg->h->step_over) { for (i = 0; i < steps; i++) if (!dbg->h->step_over (dbg)) return R_FALSE; return i; } if (!dbg->anal || !dbg->reg) return R_FALSE; // 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]); // Try to keep the buffer full if (pc - buf_pc > sizeof (buf)) { 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); return R_FALSE; } // 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)) { eprintf ("Could not step over call @ 0x%"PFMT64x"\n", pc); return R_FALSE; } } else r_debug_step (dbg, 1); } return i; } R_API int r_debug_continue_kill(RDebug *dbg, int sig) { int ret = R_FALSE; if (!dbg) return R_FALSE; if (r_debug_is_dead (dbg)) return R_FALSE; if (dbg->h && dbg->h->cont) { r_bp_restore (dbg->bp, R_TRUE); // set sw breakpoints ret = dbg->h->cont (dbg, dbg->pid, dbg->tid, sig); dbg->signum = 0; r_debug_wait (dbg); r_bp_restore (dbg->bp, R_FALSE); // unset sw breakpoints r_debug_recoil (dbg); #if 0 #if __UNIX__ /* XXX Uh? */ if (dbg->stop_all_threads && dbg->pid>0) r_sandbox_kill (dbg->pid, SIGSTOP); #endif #endif r_debug_select (dbg, dbg->pid, ret); } return ret; } R_API int r_debug_continue(RDebug *dbg) { return r_debug_continue_kill (dbg, 0); //dbg->signum); } R_API int r_debug_continue_until_nontraced(RDebug *dbg) { eprintf ("TODO\n"); return R_FALSE; } R_API int r_debug_continue_until_optype(RDebug *dbg, int type, int over) { int ret, n = 0; ut64 pc, buf_pc = 0; RAnalOp op; ut8 buf[DBG_BUF_SIZE]; if (r_debug_is_dead (dbg)) return R_FALSE; if (!dbg->anal || !dbg->reg) { eprintf ("Undefined pointer at dbg->anal\n"); return R_FALSE; } // 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 (;;) { pc = r_debug_reg_get (dbg, dbg->reg->name[R_REG_NAME_PC]); // Try to keep the buffer full if (pc - buf_pc > sizeof (buf)) { 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); return R_FALSE; } if (op.type == type) 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; } n++; } return n; } R_API int r_debug_continue_until(RDebug *dbg, ut64 addr) { int has_bp; ut64 pc; if (r_debug_is_dead (dbg)) return R_FALSE; // Check if there was another breakpoint set at addr has_bp = r_bp_at_addr (dbg->bp, addr, R_BP_PROT_EXEC) != NULL; if (!has_bp) r_bp_add_sw (dbg->bp, addr, 1, R_BP_PROT_EXEC); // 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; r_debug_continue (dbg); } // Clean up if needed if (has_bp) r_bp_del (dbg->bp, addr); return R_TRUE; } R_API int r_debug_continue_syscalls(RDebug *dbg, int *sc, int n_sc) { const char *sysname; int i, reg, ret = R_FALSE; if (!dbg || !dbg->h || r_debug_is_dead (dbg)) return R_FALSE; if (!dbg->h->contsc) { /* user-level syscall tracing */ r_debug_continue_until_optype (dbg, R_ANAL_OP_TYPE_SWI, 0); reg = (int)r_debug_reg_get (dbg, "a0"); // XXX sysname = r_syscall_get_i (dbg->anal->syscall, reg, -1); if (!sysname) sysname = "unknown"; eprintf ("--> syscall %d %s\n", reg, sysname); return reg; } if (!r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_FALSE)) { eprintf ("--> cannot read registers\n"); return -1; } reg = (int)r_debug_reg_get (dbg, "sn"); if (reg == (int)UT64_MAX) { eprintf ("Cannot find 'sn' register for current arch-os.\n"); return -1; } for (;;) { dbg->h->contsc (dbg, dbg->pid, 0); // TODO handle return value if (!r_debug_reg_sync (dbg, R_REG_TYPE_GPR, R_FALSE)) { eprintf ("--> eol\n"); return -1; } reg = (int)r_debug_reg_get (dbg, "sn"); if (reg == (int)UT64_MAX) return -1; sysname = r_syscall_get_i (dbg->anal->syscall, reg, -1); if (!sysname) sysname = "unknown"; eprintf ("--> syscall %d %s\n", reg, sysname); for (i=0; ih->contsc) { ret = dbg->h->contsc (dbg, dbg->pid, num); } else { ret = R_TRUE; // TODO.check for num } eprintf ("TODO: show syscall information\n"); /* r2rc task? ala inject? */ return ret; } R_API int r_debug_kill(RDebug *dbg, int pid, int tid, int sig) { int ret = R_FALSE; if (r_debug_is_dead (dbg)) return R_FALSE; if (dbg->h && dbg->h->kill) ret = dbg->h->kill (dbg, pid, tid, sig); else eprintf ("Backend does not implements kill()\n"); return ret; } R_API RList *r_debug_frames (RDebug *dbg, ut64 at) { if (dbg && dbg->h && dbg->h->frames) return dbg->h->frames (dbg, at); return NULL; } /* TODO: Implement fork and clone */ R_API int r_debug_child_fork (RDebug *dbg) { //if (dbg && dbg->h && dbg->h->frames) //return dbg->h->frames (dbg); return 0; } R_API int r_debug_child_clone (RDebug *dbg) { //if (dbg && dbg->h && dbg->h->frames) //return dbg->h->frames (dbg); return 0; } R_API int r_debug_is_dead (RDebug *dbg) { return (dbg->pid == -1); } R_API int r_debug_map_protect (RDebug *dbg, ut64 addr, int size, int perms) { if (dbg && dbg->h && dbg->h->map_protect) return dbg->h->map_protect (dbg, addr, size, perms); return R_FALSE; } R_API void r_debug_drx_list (RDebug *dbg) { if (dbg && dbg->h && dbg->h->drx) dbg->h->drx (dbg, 0, 0, 0, 0, 0); } R_API int r_debug_drx_set (RDebug *dbg, int idx, ut64 addr, int len, int rwx, int g) { if (dbg && dbg->h && dbg->h->drx) return dbg->h->drx (dbg, idx, addr, len, rwx, g); return R_FALSE; } R_API int r_debug_drx_unset (RDebug *dbg, int idx) { if (dbg && dbg->h && dbg->h->drx) return dbg->h->drx (dbg, idx, 0, -1, 0, 0); return R_FALSE; }