mirror of
https://github.com/radareorg/radare2.git
synced 2025-02-04 12:36:30 +00:00
36771ea713
* Fix div0 bug in r_bp_getbytes() * Show 'enabled' / 'disabled' attribute of breakpoints * Implement dbg_read and dbg_write callbacks for r_debug - Added r_debug_set_io() to setup the IO for r_debug * 'db' command can now add and remove breakpoints - Not yet written into memory. only managed
216 lines
4.3 KiB
C
216 lines
4.3 KiB
C
/* radare - LGPL - Copyright 2009 pancake<nopcode.org> */
|
|
|
|
#include <r_debug.h>
|
|
|
|
R_API int r_debug_init(struct r_debug_t *dbg)
|
|
{
|
|
dbg->pid = -1;
|
|
dbg->tid = -1;
|
|
dbg->swstep = 0; // software step
|
|
dbg->h = NULL;
|
|
r_debug_handle_init(dbg);
|
|
r_bp_init(&dbg->bp);
|
|
return R_TRUE;
|
|
}
|
|
|
|
R_API int r_debug_set_io(struct r_debug_t *dbg,
|
|
int (*_read)(void *user, int pid, u64 addr, u8 *buf, int len),
|
|
int (*_write)(void *user, int pid, u64 addr, u8 *buf, int len),
|
|
void *user)
|
|
{
|
|
dbg->read = _read;
|
|
dbg->write = _write;
|
|
dbg->user = user;
|
|
}
|
|
|
|
R_API struct r_debug_t *r_debug_new()
|
|
{
|
|
struct r_debug_t *dbg;
|
|
dbg = MALLOC_STRUCT(struct r_debug_t);
|
|
r_debug_init(dbg);
|
|
return dbg;
|
|
}
|
|
|
|
R_API int r_debug_attach(struct r_debug_t *dbg, int pid)
|
|
{
|
|
int ret = R_FALSE;
|
|
if (dbg->h && dbg->h->attach) {
|
|
ret = dbg->h->attach(pid);
|
|
if (ret) {
|
|
dbg->pid = pid;
|
|
dbg->tid = pid;
|
|
} else fprintf(stderr, "Cannot attach to this pid\n");
|
|
} else fprintf(stderr, "dbg->attach = NULL\n");
|
|
return ret;
|
|
}
|
|
|
|
R_API int r_debug_startv(struct r_debug_t *dbg, int argc, char **argv)
|
|
{
|
|
return R_FALSE;
|
|
}
|
|
|
|
R_API int r_debug_start(struct r_debug_t *dbg, const char *cmd)
|
|
{
|
|
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(struct r_debug_t *dbg, int pid, int tid)
|
|
{
|
|
dbg->pid = pid;
|
|
dbg->tid = tid;
|
|
eprintf("PID: %d %d\n", pid, tid);
|
|
return R_TRUE;
|
|
}
|
|
|
|
R_API int r_debug_set_arch(struct r_debug_t *dbg, int arch)
|
|
{
|
|
switch(arch) {
|
|
case R_DBG_ARCH_BF:
|
|
// TODO: set callbacks for brainfuck debugger here
|
|
break;
|
|
case R_DBG_ARCH_X86:
|
|
//r_reg_set(dbg->reg->nregs, R_DBG_ARCH_X86);
|
|
break;
|
|
}
|
|
return R_TRUE;
|
|
}
|
|
|
|
/*--*/
|
|
R_API int r_debug_stop_reason(struct r_debug_t *dbg)
|
|
{
|
|
// TODO: return reason to stop debugging
|
|
// - new process
|
|
// - trap instruction
|
|
// - illegal instruction
|
|
// - fpu exception
|
|
// ...
|
|
return R_TRUE;
|
|
}
|
|
|
|
R_API int r_debug_wait(struct r_debug_t *dbg)
|
|
{
|
|
int ret = R_FALSE;
|
|
if (dbg->h->wait)
|
|
ret = dbg->h->wait(dbg->pid);
|
|
return ret;
|
|
}
|
|
|
|
// TODO: count number of steps done to check if no error??
|
|
R_API int r_debug_step(struct r_debug_t *dbg, int steps)
|
|
{
|
|
int i, ret = R_FALSE;
|
|
if (dbg->h && dbg->h->step) {
|
|
for(i=0;i<steps;i++) {
|
|
ret = dbg->h->step(dbg->pid);
|
|
if (ret == R_FALSE)
|
|
break;
|
|
r_debug_wait(dbg);
|
|
// TODO: create wrapper for dbg_wait
|
|
// TODO: check return value of wait and show error
|
|
dbg->steps++;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
R_API int r_debug_syscall(struct r_debug_t *dbg, int num)
|
|
{
|
|
fprintf(stderr, "TODO\n");
|
|
return R_FALSE;
|
|
}
|
|
|
|
R_API int r_debug_step_over(struct r_debug_t *dbg, int steps)
|
|
{
|
|
// TODO: analyze opcode if it is stepoverable
|
|
fprintf(stderr, "TODO\n");
|
|
return r_debug_step(dbg, steps);
|
|
}
|
|
|
|
R_API int r_debug_continue(struct r_debug_t *dbg)
|
|
{
|
|
int ret = R_FALSE;
|
|
if (dbg->h){
|
|
if (dbg->h->cont) {
|
|
ret = dbg->h->cont(dbg->pid);
|
|
if (dbg->h->wait)
|
|
ret = dbg->h->wait(dbg->pid);
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
R_API int r_debug_continue_until(struct r_debug_t *dbg, u64 addr)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
R_API int r_debug_continue_syscall(struct r_debug_t *dbg, int sc)
|
|
{
|
|
int ret = R_FALSE;
|
|
if (dbg->h && dbg->h->contsc)
|
|
ret = dbg->h->contsc(dbg->pid, sc);
|
|
return ret;
|
|
}
|
|
|
|
// XXX wrong function name
|
|
R_API int r_debug_use_software_steps(struct r_debug_t *dbg, int value)
|
|
{
|
|
/* use software breakpoints and continues */
|
|
return -1;
|
|
}
|
|
|
|
/* registers */
|
|
|
|
R_API int r_debug_register_get(struct r_debug_t *dbg, int reg, u64 *value)
|
|
{
|
|
return R_TRUE;
|
|
}
|
|
|
|
R_API int r_debug_register_set(struct r_debug_t *dbg, int reg, u64 value)
|
|
{
|
|
return R_TRUE;
|
|
}
|
|
|
|
/* for debugging purposes? */
|
|
R_API int r_debug_register_list(struct r_debug_t *dbg)
|
|
{
|
|
int i =0;
|
|
for(i=0;i<dbg->reg.nregs;i++) {
|
|
u64 value;
|
|
r_debug_register_get(dbg, i, &value);
|
|
printf("%d %s 0x%08llx\n", i, dbg->reg.regs[i], value);
|
|
}
|
|
return R_TRUE;
|
|
}
|
|
|
|
// TODO: Move to pid.c ?
|
|
// TODO: Do we need an intermediate signal representation for portability?
|
|
// TODO: STOP, CONTINUE, KILL, ...
|
|
R_API int r_debug_kill(struct r_debug_t *dbg, int pid, int sig)
|
|
{
|
|
// XXX: use debugger handler backend here
|
|
#include <signal.h>
|
|
int ret = kill(pid, sig);
|
|
if (ret == -1)
|
|
return R_FALSE;
|
|
return R_TRUE;
|
|
}
|
|
|
|
/* mmu */
|
|
R_API int r_debug_mmu_alloc(struct r_debug_t *dbg, u64 size, u64 *addr)
|
|
{
|
|
return R_TRUE;
|
|
}
|
|
|
|
R_API int r_debug_mmu_free(struct r_debug_t *dbg, u64 addr)
|
|
{
|
|
return R_TRUE;
|
|
}
|