mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-07 22:03:13 +00:00
237 lines
5.4 KiB
C
237 lines
5.4 KiB
C
// Copyright (c) 2014-2015, The Lemon Man, All rights reserved.
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
// License as published by the Free Software Foundation; either
|
|
// version 3.0 of the License, or (at your option) any later version.
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
// Lesser General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
// License along with this library.
|
|
|
|
#include <r_asm.h>
|
|
#include <r_debug.h>
|
|
#include <wind.h>
|
|
#include <kd.h>
|
|
|
|
static WindCtx *wctx = NULL;
|
|
|
|
static int r_debug_wind_step (RDebug *dbg) {
|
|
return true;
|
|
}
|
|
|
|
static int r_debug_wind_reg_read (RDebug *dbg, int type, ut8 *buf, int size) {
|
|
(void)type;
|
|
int ret = wind_read_reg(wctx, buf, size);
|
|
if (!ret || size != ret)
|
|
return -1;
|
|
r_reg_read_regs (dbg->reg, buf, ret);
|
|
// Report as if no register has been written as we've already updated the arena here
|
|
return 0;
|
|
}
|
|
|
|
static int r_debug_wind_reg_write(RDebug *dbg, int type, const ut8 *buf, int size) {
|
|
(void)buf;
|
|
(void)size;
|
|
ut8 *arena;
|
|
int arena_size;
|
|
int ret;
|
|
|
|
if (!dbg->reg)
|
|
return false;
|
|
|
|
arena = r_reg_get_bytes (dbg->reg, R_REG_TYPE_ALL, &arena_size);
|
|
if (!arena) {
|
|
eprintf ("Could not retrieve the register arena!\n");
|
|
return false;
|
|
}
|
|
|
|
ret = wind_write_reg(wctx, arena, arena_size);
|
|
|
|
free (arena);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int r_debug_wind_continue(RDebug *dbg, int pid, int tid, int sig) {
|
|
return wind_continue(wctx);
|
|
}
|
|
|
|
static int dbreak=0;
|
|
|
|
static void wstatic_debug_break(void *u) {
|
|
dbreak = 1;
|
|
wind_break_read (wctx);
|
|
}
|
|
|
|
static int r_debug_wind_wait (RDebug *dbg, int pid) {
|
|
# define STATE_EXCEPTION 0x3030
|
|
kd_packet_t *pkt;
|
|
kd_stc_64 *stc;
|
|
int ret;
|
|
r_cons_break (wstatic_debug_break, dbg);
|
|
dbreak = 0;
|
|
for (;;) {
|
|
ret = wind_wait_packet (wctx, KD_PACKET_TYPE_STATE_CHANGE, &pkt);
|
|
if (dbreak) {
|
|
dbreak = 0;
|
|
wind_break (wctx);
|
|
continue;
|
|
}
|
|
if (ret != KD_E_OK || !pkt)
|
|
break;
|
|
stc = (kd_stc_64 *)pkt->data;
|
|
// Handle exceptions only
|
|
if (stc->state == STATE_EXCEPTION) {
|
|
wind_set_cpu (wctx, stc->cpu);
|
|
dbg->reason.type = R_DEBUG_REASON_INT;
|
|
dbg->reason.addr = stc->pc;
|
|
dbg->reason.tid = stc->kthread;
|
|
dbg->reason.signum = stc->state;
|
|
free (pkt);
|
|
break;
|
|
} else wind_continue (wctx);
|
|
free (pkt);
|
|
}
|
|
// TODO : Set the faulty process as target
|
|
|
|
return true;
|
|
}
|
|
|
|
static int r_debug_wind_attach (RDebug *dbg, int pid) {
|
|
RIODesc *desc = dbg->iob.io->desc;
|
|
|
|
if (!desc || !desc->plugin || !desc->plugin->name || !desc->data)
|
|
return false;
|
|
|
|
if (strncmp (desc->plugin->name, "windbg", 6))
|
|
return false;
|
|
|
|
if (dbg->arch && strcmp (dbg->arch, "x86"))
|
|
return false;
|
|
|
|
wctx = (WindCtx *)desc->data;
|
|
|
|
if (!wctx)
|
|
return false;
|
|
|
|
// Handshake
|
|
if (!wind_sync(wctx)) {
|
|
eprintf("Could not connect to windbg\n");
|
|
wind_ctx_free(wctx);
|
|
return false;
|
|
}
|
|
|
|
if (!wind_read_ver(wctx)) {
|
|
wind_ctx_free(wctx);
|
|
return false;
|
|
}
|
|
// Make r_debug_is_dead happy
|
|
dbg->pid = 0;
|
|
return true;
|
|
}
|
|
|
|
static int r_debug_wind_detach (RDebug *dbg, int pid) {
|
|
return true;
|
|
}
|
|
|
|
static char *r_debug_wind_reg_profile(RDebug *dbg) {
|
|
if (!dbg) return NULL;
|
|
if (dbg->arch && strcmp (dbg->arch, "x86"))
|
|
return NULL;
|
|
if (dbg->bits == R_SYS_BITS_32) {
|
|
#include "native/reg/windows-x86.h"
|
|
} else if (dbg->bits == R_SYS_BITS_64) {
|
|
#include "native/reg/windows-x64.h"
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static int r_debug_wind_breakpoint (RBreakpointItem *bp, int set, void *user) {
|
|
int *tag;
|
|
if (!bp) return false;
|
|
// Use a 32 bit word here to keep this compatible with 32 bit hosts
|
|
tag = (int *)&bp->data;
|
|
return wind_bkpt (wctx, bp->addr, set, bp->hw, tag);
|
|
}
|
|
|
|
static int r_debug_wind_init(RDebug *dbg) {
|
|
return true;
|
|
}
|
|
|
|
static RList *r_debug_wind_pids (int pid) {
|
|
RList *ret, *pids;
|
|
RListIter *it;
|
|
WindProc *p;
|
|
|
|
ret = r_list_newf (free);
|
|
if (!ret) return NULL;
|
|
|
|
pids = wind_list_process(wctx);
|
|
if (!pids)
|
|
return ret;
|
|
|
|
r_list_foreach(pids, it, p) {
|
|
RDebugPid *newpid = R_NEW0 (RDebugPid);
|
|
if (!newpid) {
|
|
r_list_free (ret);
|
|
return NULL;
|
|
}
|
|
newpid->path = strdup (p->name);
|
|
newpid->pid = p->uniqueid;
|
|
newpid->status = 's';
|
|
newpid->runnable = true;
|
|
r_list_append (ret, newpid);
|
|
}
|
|
// r_list_free (pids);
|
|
return ret;
|
|
}
|
|
|
|
static int r_debug_wind_select (int pid, int tid) {
|
|
ut64 base;
|
|
ut32 old = wind_get_target (wctx);
|
|
int ret = wind_set_target (wctx, pid);
|
|
if (!ret)
|
|
return false;
|
|
|
|
base = wind_get_target_base (wctx);
|
|
if (!base) {
|
|
wind_set_target (wctx, old);
|
|
return false;
|
|
}
|
|
|
|
eprintf ("Process base is 0x%"PFMT64x"\n", base);
|
|
return true;
|
|
}
|
|
|
|
struct r_debug_plugin_t r_debug_plugin_wind = {
|
|
.name = "wind",
|
|
.license = "LGPL3",
|
|
.arch = "x86",
|
|
.bits = R_SYS_BITS_32 | R_SYS_BITS_64,
|
|
.pids = r_debug_wind_pids,
|
|
.select = r_debug_wind_select,
|
|
.step = r_debug_wind_step,
|
|
.init = r_debug_wind_init,
|
|
.cont = r_debug_wind_continue,
|
|
.attach = &r_debug_wind_attach,
|
|
.detach = &r_debug_wind_detach,
|
|
.wait = &r_debug_wind_wait,
|
|
.breakpoint = &r_debug_wind_breakpoint,
|
|
.reg_read = &r_debug_wind_reg_read,
|
|
.reg_write = &r_debug_wind_reg_write,
|
|
.reg_profile = r_debug_wind_reg_profile,
|
|
};
|
|
|
|
#ifndef CORELIB
|
|
struct r_lib_struct_t radare_plugin = {
|
|
.type = R_LIB_TYPE_DBG,
|
|
.data = &r_debug_plugin_wind,
|
|
.version = R2_VERSION
|
|
};
|
|
#endif
|