radare2/libr/bp/bp.c
2022-08-18 10:27:00 +02:00

416 lines
9.6 KiB
C

/* radare2 - LGPL - Copyright 2009-2020 - pancake */
#include <r_bp.h>
#include <config.h>
R_LIB_VERSION (r_bp);
static struct r_bp_plugin_t *bp_static_plugins[] =
{ R_BP_STATIC_PLUGINS };
static void r_bp_item_free(RBreakpointItem *b) {
free (b->name);
free (b->bbytes);
free (b->obytes);
free (b->module_name);
free (b->data);
free (b->cond);
free (b);
}
R_API RBreakpoint *r_bp_new(void) {
int i;
RBreakpointPlugin *static_plugin;
RBreakpoint *bp = R_NEW0 (RBreakpoint);
if (!bp) {
return NULL;
}
bp->bps_idx_count = 16;
bp->bps_idx = R_NEWS0 (RBreakpointItem*, bp->bps_idx_count);
bp->stepcont = R_BP_CONT_NORMAL;
bp->traces = r_bp_traptrace_new ();
bp->cb_printf = (PrintfCallback)printf;
bp->bps = r_list_newf ((RListFree)r_bp_item_free);
bp->plugins = r_list_newf ((RListFree)free);
bp->nhwbps = 0;
for (i = 0; bp_static_plugins[i]; i++) {
static_plugin = R_NEW (RBreakpointPlugin);
memcpy (static_plugin, bp_static_plugins[i],
sizeof (RBreakpointPlugin));
r_bp_plugin_add (bp, static_plugin);
}
memset (&bp->iob, 0, sizeof (bp->iob));
return bp;
}
R_API RBreakpoint *r_bp_free(RBreakpoint *bp) {
r_list_free (bp->bps);
r_list_free (bp->plugins);
r_list_free (bp->traces);
free (bp->bps_idx);
free (bp);
return NULL;
}
R_API int r_bp_get_bytes(RBreakpoint *bp, ut8 *buf, int len, int endian, int idx) {
int i;
struct r_bp_arch_t *b;
if (bp->cur) {
// find matching size breakpoint
repeat:
for (i = 0; i < bp->cur->nbps; i++) {
b = &bp->cur->bps[i];
if (bp->cur->bps[i].bits) {
if (bp->bits != bp->cur->bps[i].bits) {
continue;
}
}
if (bp->cur->bps[i].length == len && bp->cur->bps[i].endian == endian) {
memcpy (buf, b->bytes, b->length);
return b->length;
}
}
if (len != 4) {
len = 4;
goto repeat;
}
/* if not found try to pad with the first one */
b = &bp->cur->bps[0];
if (len % b->length) {
R_LOG_WARN ("No matching bpsize");
return 0;
}
for (i = 0; i < len; i++) {
memcpy (buf + i, b->bytes, b->length);
}
return b->length;
}
return 0;
}
R_API RBreakpointItem *r_bp_get_at(RBreakpoint *bp, ut64 addr) {
RListIter *iter;
RBreakpointItem *b;
r_list_foreach(bp->bps, iter, b) {
if (b->addr == addr) {
return b;
}
}
return NULL;
}
static inline bool inRange(RBreakpointItem *b, ut64 addr) {
return (addr >= b->addr && addr < (b->addr + b->size));
}
static inline bool matchProt(RBreakpointItem *b, int perm) {
return (!perm || (perm && b->perm));
}
R_API RBreakpointItem *r_bp_get_in(RBreakpoint *bp, ut64 addr, int perm) {
RBreakpointItem *b;
RListIter *iter;
r_list_foreach (bp->bps, iter, b) {
// Check addr within range and provided perm matches (or null)
if (inRange (b, addr) && matchProt (b, perm)) {
return b;
}
}
return NULL;
}
R_API RBreakpointItem *r_bp_enable(RBreakpoint *bp, ut64 addr, int set, int count) {
RBreakpointItem *b = r_bp_get_in (bp, addr, 0);
if (b) {
b->enabled = set;
b->togglehits = count;
return b;
}
return NULL;
}
R_API int r_bp_enable_all(RBreakpoint *bp, int set) {
RListIter *iter;
RBreakpointItem *b;
r_list_foreach (bp->bps, iter, b) {
b->enabled = set;
}
return true;
}
R_API int r_bp_stepy_continuation(RBreakpoint *bp) {
// TODO: implement
return bp->stepcont;
}
static void unlinkBreakpoint(RBreakpoint *bp, RBreakpointItem *b) {
int i;
for (i = 0; i < bp->bps_idx_count; i++) {
if (bp->bps_idx[i] == b) {
bp->bps_idx[i] = NULL;
}
}
r_list_delete_data (bp->bps, b);
}
/* TODO: detect overlapping of breakpoints */
static RBreakpointItem *r_bp_add(RBreakpoint *bp, const ut8 *obytes, ut64 addr, int size, int hw, int perm) {
int ret;
RBreakpointItem *b;
if (addr == UT64_MAX || size < 1) {
return NULL;
}
if (r_bp_get_in (bp, addr, perm)) {
R_LOG_WARN ("Breakpoint already set at this address");
return NULL;
}
b = r_bp_item_new (bp);
if (!b) {
return NULL;
}
b->addr = addr + bp->delta;
if (bp->baddr > addr) {
R_LOG_WARN ("base addr should not be larger than the breakpoint address");
}
if (bp->bpinmaps && !r_bp_is_valid (bp, b)) {
R_LOG_WARN ("Cannot set breakpoint outside maps. Use dbg.bpinmaps to false");
}
b->delta = addr - bp->baddr;
b->size = size;
b->enabled = true;
b->perm = perm;
b->hw = hw;
// NOTE: for hw breakpoints there are no bytes to save/restore
if (!hw) {
b->bbytes = calloc (size + 16, 1);
if (!b->bbytes) {
return NULL;
}
if (obytes) {
b->obytes = malloc (size);
if (!b->obytes) {
free (b->bbytes);
return NULL;
}
memcpy (b->obytes, obytes, size);
} else {
b->obytes = NULL;
}
ret = r_bp_get_bytes (bp, b->bbytes, size, bp->endian, 0);
if (ret != size) {
R_LOG_WARN ("Cannot get breakpoint bytes. No architecture selected?");
}
}
bp->nbps++;
r_list_append (bp->bps, b);
return b;
}
R_API int r_bp_add_fault(RBreakpoint *bp, ut64 addr, int size, int perm) {
// TODO
return false;
}
R_API RBreakpointItem* r_bp_add_sw(RBreakpoint *bp, ut64 addr, int size, int perm) {
r_return_val_if_fail (bp && bp->iob.read_at, NULL);
if (size < 1) {
size = 1;
}
ut8 *bytes = calloc (1, size);
if (!bytes) {
return NULL;
}
bp->iob.read_at (bp->iob.io, addr, bytes, size);
RBreakpointItem *item = r_bp_add (bp, bytes, addr, size, R_BP_TYPE_SW, perm);
free (bytes);
return item;
}
R_API RBreakpointItem* r_bp_add_hw(RBreakpoint *bp, ut64 addr, int size, int perm) {
return r_bp_add (bp, NULL, addr, size, R_BP_TYPE_HW, perm);
}
// R2_580 bool
R_API int r_bp_del_all(RBreakpoint *bp) {
int i;
if (!r_list_empty (bp->bps)) {
r_list_purge (bp->bps);
for (i = 0; i < bp->bps_idx_count; i++) {
bp->bps_idx[i] = NULL;
}
return true;
}
return false;
}
// R2_580 bool
R_API int r_bp_del(RBreakpoint *bp, ut64 addr) {
RListIter *iter;
RBreakpointItem *b;
/* No _safe loop necessary because we return immediately after the delete. */
r_list_foreach (bp->bps, iter, b) {
if (b->addr == addr) {
unlinkBreakpoint (bp, b);
// r_list_delete (bp->bps, iter);
return true;
}
}
return false;
}
// R2_580 bool
R_API int r_bp_set_trace(RBreakpoint *bp, ut64 addr, int set) {
RBreakpointItem *b = r_bp_get_in (bp, addr, 0);
if (b) {
b->trace = set;
return true;
}
return false;
}
// R2_580 void
R_API int r_bp_set_trace_all(RBreakpoint *bp, int set) {
RListIter *iter;
RBreakpointItem *b;
r_list_foreach (bp->bps, iter, b) {
b->trace = set;
}
return true;
}
// TODO: deprecate
R_API int r_bp_list(RBreakpoint *bp, int rad) {
int n = 0;
RBreakpointItem *b;
RListIter *iter;
PJ *pj = NULL;
if (rad == 'j') {
pj = pj_new ();
if (!pj) {
return 0;
}
pj_a (pj);
}
r_list_foreach (bp->bps, iter, b) {
if (pj) {
pj_o (pj);
pj_kN (pj, "addr", b->addr);
pj_ki (pj, "size", b->size);
pj_ks (pj, "perm", r_str_rwx_i (b->perm & 7)); /* filter out R_BP_PROT_ACCESS */
pj_kb (pj, "hw", b->hw);
pj_kb (pj, "trace", b->trace);
pj_kb (pj, "enabled", b->enabled);
pj_kb (pj, "valid", r_bp_is_valid (bp, b));
pj_ks (pj, "data", r_str_get (b->data));
pj_ks (pj, "cond", r_str_get (b->cond));
pj_end (pj);
} else if (rad) {
if (b->module_name) {
bp->cb_printf ("dbm %s %"PFMT64d"\n", b->module_name, b->module_delta);
} else {
bp->cb_printf ("db 0x%08"PFMT64x"\n", b->addr);
}
} else {
bp->cb_printf ("0x%08"PFMT64x" - 0x%08"PFMT64x \
" %d %c%c%c %s %s %s %s cmd=\"%s\" cond=\"%s\" " \
"name=\"%s\" module=\"%s\"\n",
b->addr, b->addr + b->size, b->size,
((b->perm & R_BP_PROT_READ) | (b->perm & R_BP_PROT_ACCESS)) ? 'r' : '-',
((b->perm & R_BP_PROT_WRITE)| (b->perm & R_BP_PROT_ACCESS)) ? 'w' : '-',
(b->perm & R_BP_PROT_EXEC) ? 'x' : '-',
b->hw ? "hw": "sw",
b->trace ? "trace" : "break",
b->enabled ? "enabled" : "disabled",
r_bp_is_valid (bp, b) ? "valid" : "invalid",
r_str_get (b->data),
r_str_get (b->cond),
r_str_get (b->name),
r_str_get (b->module_name));
}
n++;
}
if (pj) {
pj_end (pj);
bp->cb_printf ("%s\n", pj_string (pj));
pj_free (pj);
}
return n;
}
R_API RBreakpointItem *r_bp_item_new(RBreakpoint *bp) {
int i, j;
/* find empty slot */
for (i = 0; i < bp->bps_idx_count; i++) {
if (!bp->bps_idx[i]) {
goto return_slot;
}
}
/* allocate new slot */
bp->bps_idx_count += 16; // allocate space for 16 more bps
RBreakpointItem **newbps = realloc (bp->bps_idx, bp->bps_idx_count * sizeof (RBreakpointItem*));
if (newbps) {
bp->bps_idx = newbps;
} else {
bp->bps_idx_count -= 16; // allocate space for 16 more bps
}
for (j = i; j < bp->bps_idx_count; j++) {
bp->bps_idx[j] = NULL;
}
return_slot:
/* empty slot */
return (bp->bps_idx[i] = R_NEW0 (RBreakpointItem));
}
R_API RBreakpointItem *r_bp_get_index(RBreakpoint *bp, int idx) {
if (idx >= 0 && idx < bp->bps_idx_count) {
return bp->bps_idx[idx];
}
return NULL;
}
R_API int r_bp_get_index_at(RBreakpoint *bp, ut64 addr) {
int i;
for (i = 0; i < bp->bps_idx_count; i++) {
if (bp->bps_idx[i] && bp->bps_idx[i]->addr == addr) {
return i;
}
}
return -1;
}
R_API int r_bp_del_index(RBreakpoint *bp, int idx) {
if (idx >= 0 && idx < bp->bps_idx_count) {
r_list_delete_data (bp->bps, bp->bps_idx[idx]);
bp->bps_idx[idx] = 0;
return true;
}
return false;
}
R_API int r_bp_size(RBreakpoint *bp) {
RBreakpointArch *bpa;
int i, bpsize = 8;
if (!bp || !bp->cur) {
return 0;
}
for (i = 0; bp->cur->bps[i].bytes; i++) {
bpa = &bp->cur->bps[i];
if (bpa->bits && bpa->bits != bp->bits) {
continue;
}
if (bpa->length < bpsize) {
bpsize = bpa->length;
}
}
return bpsize;
}
// Check if the breakpoint is in a valid map
R_API bool r_bp_is_valid(RBreakpoint *bp, RBreakpointItem *b) {
if (!bp->bpinmaps) {
return true;
}
return bp->coreb.isMapped (bp->coreb.core, b->addr, b->perm);
}