radare2/libr/debug/debug.c
pancake 36771ea713 * rasm2 -h does not list plugins (use -L)
* 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
2009-04-12 23:10:22 +00:00

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;
}