mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-25 07:15:19 +00:00
3020 lines
105 KiB
C
3020 lines
105 KiB
C
/* radare - LGPL - Copyright 2009-2018 - pancake */
|
|
|
|
#include <r_core.h>
|
|
|
|
#define NODECB(w,x,y) r_config_set_cb (cfg,w,x,y)
|
|
#define NODEICB(w,x,y) r_config_set_i_cb (cfg,w,x,y)
|
|
#define SETDESC(x,y) r_config_node_desc (x,y)
|
|
#define SETOPTIONS(x, ...) set_options (x, __VA_ARGS__)
|
|
#define SETI(x,y,z) SETDESC (r_config_set_i (cfg,x,y), z)
|
|
#define SETICB(w,x,y,z) SETDESC (NODEICB (w,x,y), z)
|
|
#define SETPREF(x,y,z) SETDESC (r_config_set (cfg,x,y), z)
|
|
#define SETCB(w,x,y,z) SETDESC (NODECB (w,x,y), z)
|
|
|
|
static void set_options(RConfigNode *node, ...) {
|
|
va_list argp;
|
|
char *option = NULL;
|
|
va_start (argp, node);
|
|
option = va_arg (argp, char *);
|
|
while (option) {
|
|
r_list_append (node->options, option);
|
|
option = va_arg (argp, char *);
|
|
}
|
|
va_end (argp);
|
|
}
|
|
|
|
static bool isGdbPlugin(RCore *core) {
|
|
if (core->io && core->io->desc && core->io->desc->plugin) {
|
|
if (core->io->desc->plugin->name && !strcmp (core->io->desc->plugin->name, "gdb")) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void print_node_options(RConfigNode *node) {
|
|
RListIter *iter;
|
|
char *option;
|
|
r_list_foreach (node->options, iter, option) {
|
|
r_cons_printf ("%s\n", option);
|
|
}
|
|
}
|
|
|
|
/* TODO: use loop here */
|
|
/*------------------------------------------------------------------------------------------*/
|
|
|
|
static int compareName(const RAnalFunction *a, const RAnalFunction *b) {
|
|
return a && b && a->name && b->name && strcmp (a->name, b->name);
|
|
}
|
|
|
|
static int compareNameLen(const RAnalFunction *a, const RAnalFunction *b) {
|
|
return a && b && a->name && b->name && strlen (a->name) > strlen (b->name);
|
|
}
|
|
|
|
static int compareAddress(const RAnalFunction *a, const RAnalFunction *b) {
|
|
return a && b && a->addr && b->addr && a->addr > b->addr;
|
|
}
|
|
|
|
static int compareType(const RAnalFunction *a, const RAnalFunction *b) {
|
|
return a && b && a->diff->type && b->diff->type && a->diff->type > b->diff->type;
|
|
}
|
|
|
|
static int compareSize(const RAnalFunction *a, const RAnalFunction *b) {
|
|
// return a && b && a->_size < b->_size;
|
|
return a && b && r_anal_fcn_realsize (a) > r_anal_fcn_realsize (b);
|
|
}
|
|
|
|
static int compareDist(const RAnalFunction *a, const RAnalFunction *b) {
|
|
return a && b && a->diff->dist && b->diff->dist && a->diff->dist > b->diff->dist;
|
|
}
|
|
|
|
static int cb_diff_sort(void *_core, void *_node) {
|
|
RConfigNode *node = _node;
|
|
const char *column = node->value;
|
|
RCore *core = _core;
|
|
if (column && strcmp (column, "?")) {
|
|
if (!strcmp (column, "name")) {
|
|
core->anal->columnSort = (RListComparator)compareName;
|
|
} else if (!strcmp (column, "namelen")) {
|
|
core->anal->columnSort = (RListComparator)compareNameLen;
|
|
} else if (!strcmp (column, "addr")) {
|
|
core->anal->columnSort = (RListComparator)compareAddress;
|
|
} else if (!strcmp (column, "type")) {
|
|
core->anal->columnSort = (RListComparator)compareType;
|
|
} else if (!strcmp (column, "size")) {
|
|
core->anal->columnSort = (RListComparator)compareSize;
|
|
} else if (!strcmp (column, "dist")) {
|
|
core->anal->columnSort = (RListComparator)compareDist;
|
|
} else {
|
|
goto fail;
|
|
}
|
|
return true;
|
|
}
|
|
fail:
|
|
eprintf ("e diff.sort = [name, namelen, addr, type, size, dist]\n");
|
|
return false;
|
|
}
|
|
|
|
static const char *has_esil(RCore *core, const char *name) {
|
|
RListIter *iter;
|
|
RAnalPlugin *h;
|
|
if (!core || !core->anal || !name) {
|
|
return NULL;
|
|
}
|
|
RAnal *a = core->anal;
|
|
r_list_foreach (a->plugins, iter, h) {
|
|
if (!strcmp (name, h->name)) {
|
|
return h->esil? "Ae": "A_";
|
|
}
|
|
}
|
|
return "__";
|
|
}
|
|
|
|
// copypasta from binr/rasm2/rasm2.c
|
|
static void rasm2_list(RCore *core, const char *arch, int fmt) {
|
|
int i;
|
|
const char *feat2, *feat;
|
|
RAsm *a = core->assembler;
|
|
char bits[32];
|
|
RAsmPlugin *h;
|
|
RListIter *iter;
|
|
if (fmt == 'j') {
|
|
r_cons_print ("{");
|
|
}
|
|
r_list_foreach (a->plugins, iter, h) {
|
|
if (arch && *arch) {
|
|
if (h->cpus && !strcmp (arch, h->name)) {
|
|
char *c = strdup (h->cpus);
|
|
int n = r_str_split (c, ',');
|
|
for (i = 0; i < n; i++) {
|
|
r_cons_println (r_str_word_get0 (c, i));
|
|
}
|
|
free (c);
|
|
break;
|
|
}
|
|
} else {
|
|
bits[0] = 0;
|
|
/* The underscore makes it easier to distinguish the
|
|
* columns */
|
|
if (h->bits&8) strcat (bits, "_8");
|
|
if (h->bits&16) strcat (bits, "_16");
|
|
if (h->bits&32) strcat (bits, "_32");
|
|
if (h->bits&64) strcat (bits, "_64");
|
|
if (!*bits) strcat (bits, "_0");
|
|
feat = "__";
|
|
if (h->assemble && h->disassemble) feat = "ad";
|
|
if (h->assemble && !h->disassemble) feat = "a_";
|
|
if (!h->assemble && h->disassemble) feat = "_d";
|
|
feat2 = has_esil (core, h->name);
|
|
if (fmt == 'q') {
|
|
r_cons_println (h->name);
|
|
} else if (fmt == 'j') {
|
|
const char *str_bits = "32, 64";
|
|
const char *license = "GPL";
|
|
r_cons_printf ("\"%s\":{\"bits\":[%s],\"license\":\"%s\",\"description\":\"%s\",\"features\":\"%s\"}%s",
|
|
h->name, str_bits, license, h->desc, feat, iter->n? ",": "");
|
|
} else {
|
|
r_cons_printf ("%s%s %-9s %-11s %-7s %s\n",
|
|
feat, feat2, bits, h->name,
|
|
h->license?h->license:"unknown", h->desc);
|
|
}
|
|
}
|
|
}
|
|
if (fmt == 'j') {
|
|
r_cons_print ("}\n");
|
|
}
|
|
}
|
|
|
|
static inline void __setsegoff(RConfig *cfg, const char *asmarch, int asmbits) {
|
|
int autoseg = (!strncmp (asmarch, "x86", 3) && asmbits == 16);
|
|
r_config_set (cfg, "asm.segoff", r_str_bool (autoseg));
|
|
}
|
|
|
|
static int cb_debug_hitinfo(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->dbg->hitinfo = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analarmthumb(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.armthumb = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analeobjmp(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.eobjmp = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analdepth(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.depth = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analafterjmp(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.afterjmp = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analvars(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.vars = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analstrings(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (node->i_value) {
|
|
r_config_set (core->config, "bin.strings", "false");
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_analsleep(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->sleep = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analmaxrefs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->maxreflines = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analnopskip (void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.nopskip = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analhpskip (void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.hpskip = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analbbsplit (void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.bbsplit = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
/* obey section permissions */
|
|
static int cb_analnoncode(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.noncode = !!node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static void update_analarch_options(RCore *core, RConfigNode *node) {
|
|
RAnalPlugin *h;
|
|
RListIter *it;
|
|
if (core && core->anal && node) {
|
|
r_list_purge (node->options);
|
|
r_list_foreach (core->anal->plugins, it, h) {
|
|
SETOPTIONS (node, h->name, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int cb_analarch(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (*node->value == '?') {
|
|
update_analarch_options (core, node);
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
if (*node->value) {
|
|
if (r_anal_use (core->anal, node->value)) {
|
|
return true;
|
|
}
|
|
const char *aa = r_config_get (core->config, "asm.arch");
|
|
if (!aa || strcmp (aa, node->value)) {
|
|
eprintf ("anal.arch: cannot find '%s'\n", node->value);
|
|
} else {
|
|
r_config_set (core->config, "anal.arch", "null");
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int cb_analcpu(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_anal_set_cpu (core->anal, node->value);
|
|
/* set pcalign */
|
|
{
|
|
int v = r_anal_archinfo (core->anal, R_ANAL_ARCHINFO_ALIGN);
|
|
r_config_set_i (core->config, "asm.pcalign", (v != -1)? v: 0);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_analsplit(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->split = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analrecont(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.recont = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_analijmp(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.ijmp = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_asmvarsubmin(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->parser->minval = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrrainbow(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->print->flags |= R_PRINT_FLAGS_RAINBOW;
|
|
} else {
|
|
core->print->flags &= (~R_PRINT_FLAGS_RAINBOW);
|
|
}
|
|
r_print_set_flags (core->print, core->print->flags);
|
|
return true;
|
|
}
|
|
|
|
static int cb_asmsecsub(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->print->flags |= R_PRINT_FLAGS_SECSUB;
|
|
} else {
|
|
core->print->flags &= (~R_PRINT_FLAGS_SECSUB);
|
|
}
|
|
r_print_set_flags (core->print, core->print->flags);
|
|
return true;
|
|
}
|
|
|
|
static int cb_asmassembler(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_asm_use_assembler (core->assembler, node->value);
|
|
return true;
|
|
}
|
|
|
|
static void update_asmcpu_options(RCore *core, RConfigNode *node) {
|
|
RAsmPlugin *h;
|
|
RListIter *iter;
|
|
if (!core || !core->assembler) {
|
|
return;
|
|
}
|
|
const char *arch = r_config_get (core->config, "asm.arch");
|
|
if (!arch || !*arch) {
|
|
return;
|
|
}
|
|
r_list_purge (node->options);
|
|
r_list_foreach (core->assembler->plugins, iter, h) {
|
|
if (h->cpus && !strcmp (arch, h->name)) {
|
|
char *c = strdup (h->cpus);
|
|
int i, n = r_str_split (c, ',');
|
|
for (i = 0; i < n; i++) {
|
|
SETOPTIONS (node, r_str_word_get0 (c, i), NULL);
|
|
}
|
|
free (c);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int cb_asmcpu(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (*node->value == '?') {
|
|
update_asmcpu_options (core, node);
|
|
/* print verbose help instead of plain option listing */
|
|
rasm2_list (core, r_config_get (core->config, "asm.arch"), node->value[1]);
|
|
return 0;
|
|
}
|
|
r_asm_set_cpu (core->assembler, node->value);
|
|
r_config_set (core->config, "anal.cpu", node->value);
|
|
return true;
|
|
}
|
|
|
|
static void update_asmarch_options(RCore *core, RConfigNode *node) {
|
|
RAsmPlugin *h;
|
|
RListIter *iter;
|
|
if (core && node && core->assembler) {
|
|
r_list_purge (node->options);
|
|
r_list_foreach (core->assembler->plugins, iter, h) {
|
|
SETOPTIONS (node, h->name, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int cb_asmarch(void *user, void *data) {
|
|
char asmparser[32];
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
const char *asmos = NULL;
|
|
int bits = R_SYS_BITS;
|
|
if (!*node->value || !core || !core->assembler) {
|
|
return false;
|
|
}
|
|
asmos = r_config_get (core->config, "asm.os");
|
|
if (core && core->anal && core->anal->bits) {
|
|
bits = core->anal->bits;
|
|
}
|
|
if (node->value[0] == '?') {
|
|
update_asmarch_options (core, node);
|
|
if (strlen (node->value) > 1 && node->value[1] == '?') {
|
|
/* print more verbose help instead of plain option values */
|
|
rasm2_list (core, NULL, node->value[1]);
|
|
return false;
|
|
} else {
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
}
|
|
r_egg_setup (core->egg, node->value, bits, 0, R_SYS_OS);
|
|
|
|
if (!r_asm_use (core->assembler, node->value)) {
|
|
eprintf ("asm.arch: cannot find (%s)\n", node->value);
|
|
return false;
|
|
}
|
|
//we should strdup here otherwise will crash if any r_config_set
|
|
//free the old value
|
|
char *asm_cpu = strdup (r_config_get (core->config, "asm.cpu"));
|
|
if (core->assembler->cur) {
|
|
const char *newAsmCPU = core->assembler->cur->cpus;
|
|
if (newAsmCPU) {
|
|
if (*newAsmCPU) {
|
|
char *nac = strdup (newAsmCPU);
|
|
char *comma = strchr (nac, ',');
|
|
if (comma) {
|
|
*comma = 0;
|
|
r_config_set (core->config, "asm.cpu", nac);
|
|
}
|
|
free (nac);
|
|
} else {
|
|
r_config_set (core->config, "asm.cpu", "");
|
|
}
|
|
}
|
|
bits = core->assembler->cur->bits;
|
|
if (8 & bits) {
|
|
bits = 8;
|
|
} else if (16 & bits) {
|
|
bits = 16;
|
|
} else if (32 & bits) {
|
|
bits = 32;
|
|
} else {
|
|
bits = 64;
|
|
}
|
|
}
|
|
snprintf (asmparser, sizeof (asmparser), "%s.pseudo", node->value);
|
|
r_config_set (core->config, "asm.parser", asmparser);
|
|
if (core->assembler->cur && core->anal &&
|
|
!(core->assembler->cur->bits & core->anal->bits)) {
|
|
r_config_set_i (core->config, "asm.bits", bits);
|
|
}
|
|
|
|
//r_debug_set_arch (core->dbg, r_sys_arch_id (node->value), bits);
|
|
r_debug_set_arch (core->dbg, node->value, bits);
|
|
if (!r_config_set (core->config, "anal.arch", node->value)) {
|
|
char *p, *s = strdup (node->value);
|
|
if (s) {
|
|
p = strchr (s, '.');
|
|
if (p) {
|
|
*p = 0;
|
|
}
|
|
if (!r_config_set (core->config, "anal.arch", s)) {
|
|
/* fall back to the anal.null plugin */
|
|
r_config_set (core->config, "anal.arch", "null");
|
|
}
|
|
free (s);
|
|
}
|
|
}
|
|
// set pcalign
|
|
if (core->anal) {
|
|
const char *asmcpu = r_config_get (core->config, "asm.cpu");
|
|
if (!r_syscall_setup (core->anal->syscall, node->value, core->anal->bits, asmcpu, asmos)) {
|
|
//eprintf ("asm.arch: Cannot setup syscall '%s/%s' from '%s'\n",
|
|
// node->value, asmos, R2_LIBDIR"/radare2/"R2_VERSION"/syscall");
|
|
}
|
|
}
|
|
//if (!strcmp (node->value, "bf"))
|
|
// r_config_set (core->config, "dbg.backend", "bf");
|
|
__setsegoff (core->config, node->value, core->assembler->bits);
|
|
|
|
// set a default endianness
|
|
int bigbin = r_bin_is_big_endian (core->bin);
|
|
if (bigbin == -1 /* error: no endianness detected in binary */) {
|
|
bigbin = r_config_get_i (core->config, "cfg.bigendian");
|
|
}
|
|
|
|
// try to set endian of RAsm to match binary
|
|
r_asm_set_big_endian (core->assembler, bigbin);
|
|
// set endian of display to match binary
|
|
core->print->big_endian = bigbin;
|
|
|
|
r_asm_set_cpu (core->assembler, asm_cpu);
|
|
free (asm_cpu);
|
|
RConfigNode *asmcpu = r_config_node_get (core->config, "asm.cpu");
|
|
if (asmcpu) {
|
|
update_asmcpu_options (core, asmcpu);
|
|
}
|
|
{
|
|
int v = r_anal_archinfo (core->anal, R_ANAL_ARCHINFO_ALIGN);
|
|
if (v != -1) {
|
|
r_config_set_i (core->config, "asm.pcalign", v);
|
|
} else {
|
|
r_config_set_i (core->config, "asm.pcalign", 0);
|
|
}
|
|
}
|
|
/* reload types and cc info */
|
|
// changing asm.arch changes anal.arch
|
|
// changing anal.arch sets types db
|
|
// so ressetting is redundant and may lead to bugs
|
|
// 1 case this is usefull is when sdb_types is null
|
|
if (!core->anal || !core->anal->sdb_types) {
|
|
r_core_anal_type_init (core);
|
|
}
|
|
r_core_anal_cc_init (core);
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbgbpsize(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->dbg->bpsize = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbgbtdepth(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->dbg->btdepth = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_asmbits(void *user, void *data) {
|
|
const char *asmos, *asmarch, *asmcpu;
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
int ret = 0, bits;
|
|
if (!core) {
|
|
eprintf ("user can't be NULL\n");
|
|
return false;
|
|
}
|
|
|
|
bits = node->i_value;
|
|
#if 0
|
|
// TODO: pretty good optimization, but breaks many tests when arch is different i think
|
|
if (bits == core->assembler->bits && bits == core->anal->bits && bits == core->dbg->bits) {
|
|
// early optimization
|
|
return true;
|
|
}
|
|
#endif
|
|
if (bits > 0) {
|
|
ret = r_asm_set_bits (core->assembler, bits);
|
|
if (!ret) {
|
|
RAsmPlugin *h = core->assembler->cur;
|
|
if (!h) {
|
|
eprintf ("e asm.bits: Cannot set value, no plugins defined yet\n");
|
|
ret = true;
|
|
}
|
|
// else { eprintf ("Cannot set bits %d to '%s'\n", bits, h->name); }
|
|
}
|
|
if (!r_anal_set_bits (core->anal, bits)) {
|
|
eprintf ("asm.arch: Cannot setup '%d' bits analysis engine\n", bits);
|
|
}
|
|
core->print->bits = bits;
|
|
}
|
|
if (core->dbg && core->anal && core->anal->cur) {
|
|
r_debug_set_arch (core->dbg, core->anal->cur->arch, bits);
|
|
bool load_from_debug = r_config_get_i (core->config, "cfg.debug");
|
|
if (load_from_debug) {
|
|
if (core->dbg->h && core->dbg->h->reg_profile) {
|
|
// XXX. that should depend on the plugin, not the host os
|
|
#if __WINDOWS__
|
|
#if !defined(__MINGW64__) && !defined(_WIN64)
|
|
core->dbg->bits = R_SYS_BITS_32;
|
|
#else
|
|
core->dbg->bits = R_SYS_BITS_64;
|
|
#endif
|
|
#endif
|
|
char *rp = core->dbg->h->reg_profile (core->dbg);
|
|
r_reg_set_profile_string (core->dbg->reg, rp);
|
|
r_reg_set_profile_string (core->anal->reg, rp);
|
|
free (rp);
|
|
}
|
|
} else {
|
|
(void)r_anal_set_reg_profile (core->anal);
|
|
}
|
|
}
|
|
asmos = r_config_get (core->config, "asm.os");
|
|
asmarch = r_config_get (core->config, "asm.arch");
|
|
asmcpu = r_config_get (core->config, "asm.cpu");
|
|
if (core->anal) {
|
|
if (!r_syscall_setup (core->anal->syscall, asmarch, bits, asmcpu, asmos)) {
|
|
//eprintf ("asm.arch: Cannot setup syscall '%s/%s' from '%s'\n",
|
|
// node->value, asmos, R2_LIBDIR"/radare2/"R2_VERSION"/syscall");
|
|
}
|
|
__setsegoff (core->config, asmarch, core->anal->bits);
|
|
if (core->dbg) {
|
|
r_bp_use (core->dbg->bp, asmarch, core->anal->bits);
|
|
r_config_set_i (core->config, "dbg.bpsize", r_bp_size (core->dbg->bp));
|
|
}
|
|
}
|
|
/* set pcalign */
|
|
{
|
|
int v = r_anal_archinfo (core->anal, R_ANAL_ARCHINFO_ALIGN);
|
|
r_config_set_i (core->config, "asm.pcalign", (v != -1)? v: 0);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static void update_asmfeatures_options(RCore *core, RConfigNode *node) {
|
|
int i, argc;
|
|
char *features;
|
|
|
|
if (core && core->assembler && core->assembler->cur) {
|
|
if (core->assembler->cur->features) {
|
|
features = strdup (core->assembler->cur->features);
|
|
argc = r_str_split (features, ',');
|
|
for (i = 0; i < argc; i++) {
|
|
node->options->free = free;
|
|
const char *feature = r_str_word_get0 (features, i);
|
|
r_list_append (node->options, strdup (feature));
|
|
}
|
|
free (features);
|
|
}
|
|
}
|
|
}
|
|
|
|
static int cb_asmfeatures(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (*node->value == '?') {
|
|
update_asmfeatures_options (core, node);
|
|
print_node_options (node);
|
|
return 0;
|
|
}
|
|
free (core->assembler->features);
|
|
core->assembler->features = NULL;
|
|
if (node->value[0]) {
|
|
core->assembler->features = strdup (node->value);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int cb_asmlineswidth(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->anal->lineswidth = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_emustr(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
r_config_set (core->config, "asm.emu", "true");
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_emuskip(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (*node->value == '?') {
|
|
if (strlen (node->value) > 1 && node->value[1] == '?') {
|
|
r_cons_printf ("Concatenation of meta types encoded as characters:\n" \
|
|
"'d': data\n'c': code\n's': string\n'f': format\n'm': magic\n" \
|
|
"'h': hide\n'C': comment\n'r': run\n" \
|
|
"(default is 'ds' to skip data and strings)\n");
|
|
} else {
|
|
print_node_options (node);
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_asm_armimm(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->assembler->immdisp = node->i_value ? true : false;
|
|
return true;
|
|
}
|
|
|
|
static int cb_asm_invhex(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->assembler->invhex = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_asm_pcalign(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
int align = node->i_value;
|
|
if (align < 0) {
|
|
align = 0;
|
|
}
|
|
core->assembler->pcalign = align;
|
|
core->anal->pcalign = align;
|
|
return true;
|
|
}
|
|
|
|
static int cb_asmos(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
int asmbits = r_config_get_i (core->config, "asm.bits");
|
|
RConfigNode *asmarch, *node = (RConfigNode*) data;
|
|
|
|
if (*node->value == '?') {
|
|
print_node_options (node);
|
|
return 0;
|
|
}
|
|
if (!node->value[0]) {
|
|
free (node->value);
|
|
node->value = strdup (R_SYS_OS);
|
|
}
|
|
asmarch = r_config_node_get (core->config, "asm.arch");
|
|
if (asmarch) {
|
|
const char *asmcpu = r_config_get (core->config, "asm.cpu");
|
|
r_syscall_setup (core->anal->syscall, asmarch->value, core->anal->bits, asmcpu, node->value);
|
|
__setsegoff (core->config, asmarch->value, asmbits);
|
|
}
|
|
r_anal_set_os (core->anal, node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_asmparser(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
return r_parse_use (core->parser, node->value);
|
|
}
|
|
|
|
static int cb_asmstrenc (void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
if (node->value[0] == '?') {
|
|
print_node_options (node);
|
|
r_cons_printf (" -- if string's 2nd & 4th bytes are 0 then utf16le else "
|
|
"if 2nd - 4th & 6th bytes are 0 & no char > 0x10ffff then utf32le else "
|
|
"if utf8 char detected then utf8 else latin1\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_binfilter(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->bin->filter = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
/* BinDemangleCmd */
|
|
static int cb_bdc(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->bin->demanglercmd = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_useldr(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->bin->use_ldr = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_usextr(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->bin->use_xtr = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_strpurge(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (*node->value == '?') {
|
|
r_cons_printf (
|
|
"There can be multiple entries separated by commas. No whitespace before/after entries.\n"
|
|
"Possible entries:\n"
|
|
" all : purge all strings\n"
|
|
" true : use the false_positive() classifier in cbin.c\n"
|
|
" addr : purge string at addr\n"
|
|
" addr1-addr2 : purge all strings in the range addr1-addr2 inclusive\n"
|
|
" !addr : prevent purge of string at addr by prev entries\n"
|
|
" !addr1-addr2 : prevent purge of strings in range addr1-addr2 inclusive by prev entries\n"
|
|
"Neither !true nor !false is supported.\n"
|
|
"\n"
|
|
"Examples:\n"
|
|
" e bin.strpurge=true,0-0xff,!0x1a\n"
|
|
" -- purge strings using the false_positive() classifier in cbin.c and also strings \n"
|
|
" with addresses in the range 0-0xff, but not the string at 0x1a.\n"
|
|
" e bin.strpurge=all,!0x1000-0x1fff\n"
|
|
" -- purge all strings except the strings with addresses in the range 0x1000-0x1fff.\n");
|
|
return false;
|
|
}
|
|
free (core->bin->strpurge);
|
|
core->bin->strpurge = !*node->value || !strcmp (node->value, "false")
|
|
? NULL : strdup (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_midflags (void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
if (node->value[0] == '?') {
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_strfilter(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (node->value[0] == '?') {
|
|
if (strlen (node->value) > 1 && node->value[1] == '?') {
|
|
r_cons_printf ("Valid values for bin.strfilter:\n"
|
|
"a only alphanumeric printable\n"
|
|
"8 only strings with utf8 chars\n"
|
|
"p file/directory paths\n"
|
|
"e email-like addresses\n"
|
|
"u urls\n"
|
|
"i IPv4 address-like strings\n"
|
|
"U only uppercase strings\n"
|
|
"f format-strings\n");
|
|
} else {
|
|
print_node_options (node);
|
|
}
|
|
return false;
|
|
} else {
|
|
core->bin->strfilter = node->value[0];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_binforce(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
r_bin_force_plugin (core->bin, node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_asmsyntax(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (*node->value == '?') {
|
|
print_node_options (node);
|
|
return false;
|
|
} else {
|
|
int syntax = r_asm_syntax_from_string (node->value);
|
|
if (syntax == -1) {
|
|
return false;
|
|
}
|
|
r_asm_set_syntax (core->assembler, syntax);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dirzigns(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
free (core->anal->zign_path);
|
|
core->anal->zign_path = strdup (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_bigendian(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
// Try to set endian based on preference, restrict by RAsmPlugin
|
|
bool isbig = r_asm_set_big_endian (core->assembler, node->i_value);
|
|
// Set anal endianness the same as asm
|
|
r_anal_set_big_endian (core->anal, isbig);
|
|
// the big endian should also be assigned to dbg->bp->endian
|
|
if (core->dbg && core->dbg->bp) {
|
|
core->dbg->bp->endian = isbig;
|
|
}
|
|
// Set printing endian to user's choice
|
|
core->print->big_endian = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_cfgdatefmt(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
snprintf (core->print->datefmt, 32, "%s", node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_timezone(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->print->datezone = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_cfglog(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->cfglog = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_cfgdebug(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (!core) {
|
|
return false;
|
|
}
|
|
if (core->io) {
|
|
core->io->debug = node->i_value;
|
|
}
|
|
if (core->dbg && node->i_value) {
|
|
const char *dbgbackend = r_config_get (core->config, "dbg.backend");
|
|
core->bin->is_debugger = true;
|
|
r_debug_use (core->dbg, dbgbackend);
|
|
if (!strcmp (r_config_get (core->config, "cmd.prompt"), "")) {
|
|
r_config_set (core->config, "cmd.prompt", ".dr*");
|
|
}
|
|
if (!strcmp (dbgbackend, "bf")) {
|
|
r_config_set (core->config, "asm.arch", "bf");
|
|
}
|
|
if (core->file) {
|
|
r_debug_select (core->dbg, r_io_fd_get_pid (core->io, core->file->fd),
|
|
r_io_fd_get_tid (core->io, core->file->fd));
|
|
}
|
|
} else {
|
|
r_debug_use (core->dbg, NULL);
|
|
core->bin->is_debugger = false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dirsrc(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
RCore *core = (RCore *)user;
|
|
free (core->bin->srcdir);
|
|
core->bin->srcdir = strdup (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_cfgsanbox(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
int ret = r_sandbox_enable (node->i_value);
|
|
if (node->i_value != ret) {
|
|
eprintf ("Cannot disable sandbox\n");
|
|
}
|
|
return (!node->i_value && ret)? 0: 1;
|
|
}
|
|
|
|
static int cb_str_escbslash(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->print->esc_bslash = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_cfg_fortunes(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
// TODO CN_BOOL option does not receive the right hand side of assignment as an argument
|
|
if (node->value[0] == '?') {
|
|
r_core_fortune_list (core);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_cfg_fortunes_type(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
if (node->value[0] == '?') {
|
|
r_core_fortune_list_types ();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmdpdc(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
if (node->value[0] == '?') {
|
|
r_cons_printf ("pdc\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmdlog(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
R_FREE (core->cmdlog);
|
|
core->cmdlog = strdup (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmdrepeat(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->cmdrepeat = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrnull(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->cons->null = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_color(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->print->flags |= R_PRINT_FLAGS_COLOR;
|
|
} else {
|
|
core->print->flags &= (~R_PRINT_FLAGS_COLOR);
|
|
}
|
|
if (!strcmp (node->value, "true")) {
|
|
node->i_value = 1;
|
|
} else if (!strcmp (node->value, "false")) {
|
|
node->i_value = 0;
|
|
}
|
|
r_cons_singleton ()->color = (node->i_value > COLOR_MODE_16M)
|
|
? COLOR_MODE_16M: node->i_value;
|
|
r_cons_pal_update_event ();
|
|
r_print_set_flags (core->print, core->print->flags);
|
|
return true;
|
|
}
|
|
|
|
static int cb_decoff(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->print->flags |= R_PRINT_FLAGS_ADDRDEC;
|
|
} else {
|
|
core->print->flags &= (~R_PRINT_FLAGS_ADDRDEC);
|
|
}
|
|
r_print_set_flags (core->print, core->print->flags);
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbgbep(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (*node->value == '?') {
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_btalgo(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (*node->value == '?') {
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
free (core->dbg->btalgo);
|
|
core->dbg->btalgo = strdup (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_libs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
free (core->dbg->glob_libs);
|
|
core->dbg->glob_libs = strdup (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_unlibs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
free (core->dbg->glob_unlibs);
|
|
core->dbg->glob_unlibs = strdup (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_forks(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->dbg->trace_forks = node->i_value;
|
|
if (core->io->debug) {
|
|
r_debug_attach (core->dbg, core->dbg->pid);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_gdb_page_size(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (node->i_value < 64) { // 64 is hardcoded min packet size
|
|
return false;
|
|
}
|
|
if (isGdbPlugin (core)) {
|
|
char cmd[64];
|
|
snprintf (cmd, sizeof (cmd), "page_size %"PFMT64d, node->i_value);
|
|
free (r_io_system (core->io, cmd));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_gdb_retries(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (node->i_value <= 0) {
|
|
return false;
|
|
}
|
|
if (isGdbPlugin (core)) {
|
|
char cmd[64];
|
|
snprintf (cmd, sizeof (cmd), "retries %"PFMT64d, node->i_value);
|
|
free (r_io_system (core->io, cmd));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_execs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->dbg->trace_execs = node->i_value;
|
|
if (core->io->debug) {
|
|
r_debug_attach (core->dbg, core->dbg->pid);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_clone(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->dbg->trace_clone = node->i_value;
|
|
if (core->io->debug) {
|
|
r_debug_attach (core->dbg, core->dbg->pid);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_follow_child(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->dbg->follow_child = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_aftersc(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->dbg->trace_aftersyscall = node->i_value;
|
|
if (core->io->debug) {
|
|
r_debug_attach (core->dbg, core->dbg->pid);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_runprofile(void *user, void *data) {
|
|
RCore *r = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
free ((void*)r->io->runprofile);
|
|
if (!node || !*(node->value)) {
|
|
r->io->runprofile = NULL;
|
|
} else {
|
|
r->io->runprofile = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbg_args(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (!node || !*(node->value)) {
|
|
core->io->args = NULL;
|
|
} else {
|
|
core->io->args = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbgstatus(void *user, void *data) {
|
|
RCore *r = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (r_config_get_i (r->config, "cfg.debug")) {
|
|
if (node->i_value) {
|
|
r_config_set (r->config, "cmd.prompt",
|
|
".dr*; drd; sr PC;pi 1;s-");
|
|
} else {
|
|
r_config_set (r->config, "cmd.prompt", ".dr*");
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbgbackend(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (!strcmp (node->value, "?")) {
|
|
r_debug_plugin_list (core->dbg, 'q');
|
|
return false;
|
|
}
|
|
if (!strcmp (node->value, "bf")) {
|
|
r_config_set (core->config, "asm.arch", "bf");
|
|
}
|
|
r_debug_use (core->dbg, node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_gotolimit(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (r_sandbox_enable (0)) {
|
|
eprintf ("Cannot change gotolimit\n");
|
|
return false;
|
|
}
|
|
if (core->anal->esil) {
|
|
core->anal->esil_goto_limit = node->i_value;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_esilverbose (void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (core->anal->esil) {
|
|
core->anal->esil->verbose = node->i_value;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_esilstackdepth (void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (node->i_value < 3) {
|
|
eprintf ("esil.stack.depth must be greater than 2\n");
|
|
node->i_value = 32;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_fixrows(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->fix_rows = (int)node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_fixcolumns(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->fix_columns = atoi (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_rows(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->force_rows = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_hexcompact(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->print->flags |= R_PRINT_FLAGS_COMPACT;
|
|
} else {
|
|
core->print->flags &= (~R_PRINT_FLAGS_COMPACT);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_hexpairs(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->print->pairs = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_hexcomments(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->print->flags |= R_PRINT_FLAGS_COMMENT;
|
|
} else {
|
|
core->print->flags &= ~R_PRINT_FLAGS_COMMENT;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_iopcache(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if ((bool)node->i_value) {
|
|
if (core) {
|
|
r_config_set_i (core->config, "io.pcache.read", true);
|
|
r_config_set_i (core->config, "io.pcache.write", true);
|
|
}
|
|
} else {
|
|
if (core && core->io) {
|
|
r_io_desc_cache_fini_all (core->io);
|
|
r_config_set_i (core->config, "io.pcache.read", false);
|
|
r_config_set_i (core->config, "io.pcache.write", false);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_iopcacheread(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if ((bool)node->i_value) {
|
|
if (core && core->io) {
|
|
core->io->p_cache |= 1;
|
|
}
|
|
} else {
|
|
if (core && core->io && core->io->p_cache) {
|
|
core->io->p_cache &= 2;
|
|
if (!(core->io->p_cache & 2)) {
|
|
r_io_desc_cache_fini_all (core->io);
|
|
r_config_set_i (core->config, "io.pcache", false);
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_iopcachewrite(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if ((bool)node->i_value) {
|
|
if (core && core->io) {
|
|
core->io->p_cache |= 2;
|
|
}
|
|
} else {
|
|
if (core && core->io && core->io->p_cache) {
|
|
core->io->p_cache &= 1;
|
|
if (!(core->io->p_cache & 1)) {
|
|
r_io_desc_cache_fini_all (core->io);
|
|
r_config_set_i (core->config, "io.pcache", false);
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API bool r_core_esil_cmd(RAnalEsil *esil, const char *cmd, ut64 a1, ut64 a2) {
|
|
if (cmd && *cmd) {
|
|
RCore *core = esil->anal->user;
|
|
r_core_cmdf (core, "%s %"PFMT64d" %" PFMT64d, cmd, a1, a2);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int cb_cmd_esil_ioer(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core && core->anal && core->anal->esil) {
|
|
core->anal->esil->cmd = r_core_esil_cmd;
|
|
free (core->anal->esil->cmd_ioer);
|
|
core->anal->esil->cmd_ioer = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmd_esil_todo(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core && core->anal && core->anal->esil) {
|
|
core->anal->esil->cmd = r_core_esil_cmd;
|
|
free (core->anal->esil->cmd_todo);
|
|
core->anal->esil->cmd_todo = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmd_esil_intr(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core && core->anal && core->anal->esil) {
|
|
core->anal->esil->cmd = r_core_esil_cmd;
|
|
free (core->anal->esil->cmd_intr);
|
|
core->anal->esil->cmd_intr = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_mdevrange(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core && core->anal && core->anal->esil) {
|
|
core->anal->esil->cmd = r_core_esil_cmd;
|
|
free (core->anal->esil->mdev_range);
|
|
core->anal->esil->mdev_range = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmd_esil_mdev(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core && core->anal && core->anal->esil) {
|
|
core->anal->esil->cmd = r_core_esil_cmd;
|
|
free (core->anal->esil->cmd_mdev);
|
|
core->anal->esil->cmd_mdev = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmd_esil_trap(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core && core->anal && core->anal->esil) {
|
|
core->anal->esil->cmd = r_core_esil_cmd;
|
|
core->anal->esil->cmd_trap = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_fsview(void *user, void *data) {
|
|
int type = R_FS_VIEW_NORMAL;
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (*node->value == '?') {
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
if (!strcmp (node->value, "all")) {
|
|
type = R_FS_VIEW_ALL;
|
|
}
|
|
if (!strstr (node->value, "del")) {
|
|
type |= R_FS_VIEW_DELETED;
|
|
}
|
|
if (!strstr (node->value, "spe")) {
|
|
type |= R_FS_VIEW_SPECIAL;
|
|
}
|
|
r_fs_view (core->fs, type);
|
|
return true;
|
|
}
|
|
|
|
static int cb_cmddepth(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
int c = R_MAX (((RConfigNode*)data)->i_value, 0);
|
|
core->max_cmd_depth = c;
|
|
core->cmd_depth = c;
|
|
return true;
|
|
}
|
|
|
|
static int cb_hexcols(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
int c = R_MIN (1024, R_MAX (((RConfigNode*)data)->i_value, 0));
|
|
if (c < 0) {
|
|
c = 0;
|
|
}
|
|
core->print->cols = c & ~1;
|
|
core->dbg->regcols = c/4;
|
|
return true;
|
|
}
|
|
|
|
static int cb_hexstride(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
((RCore *)user)->print->stride = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_search_kwidx(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->search->n_kws = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_iobuffer(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
ut64 from, to;
|
|
from = r_config_get_i (core->config, "io.buffer.from");
|
|
to = r_config_get_i (core->config, "io.buffer.to");
|
|
if (from>=to) {
|
|
eprintf ("ERROR: io.buffer.from >= io.buffer.to"
|
|
" (0x%"PFMT64x" >= 0x%"PFMT64x")\n", from, to);
|
|
} else {
|
|
r_io_buffer_load (core->io, from, (int)(to-from));
|
|
}
|
|
} else {
|
|
r_io_buffer_close (core->io);
|
|
}
|
|
r_core_block_read (core);
|
|
return true;
|
|
}
|
|
|
|
static int cb_io_cache_mode(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
if (node->i_value) {
|
|
core->io->cachemode = true;
|
|
} else {
|
|
core->io->cachemode = false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_io_cache_read(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
if (node->i_value) {
|
|
core->io->cached |= R_IO_READ;
|
|
} else {
|
|
core->io->cached &= ~R_IO_READ;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_io_cache_write(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
RConfigNode *node = (RConfigNode *)data;
|
|
if (node->i_value) {
|
|
core->io->cached |= R_IO_WRITE;
|
|
} else {
|
|
core->io->cached &= ~R_IO_WRITE;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_io_cache(void *user, void *data) {
|
|
(void)cb_io_cache_read (user, data);
|
|
(void)cb_io_cache_write (user, data);
|
|
return true;
|
|
}
|
|
|
|
static int cb_ioaslr(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value != core->io->aslr) {
|
|
core->io->aslr = node->i_value;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_iova(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value != core->io->va) {
|
|
core->io->va = node->i_value;
|
|
/* ugly fix for r2 -d ... "r2 is going to die soon ..." */
|
|
if (core->io->desc) {
|
|
r_core_block_read (core);
|
|
}
|
|
#if 0
|
|
/* reload symbol information */
|
|
if (r_list_length (r_bin_get_sections (core->bin)) > 0) {
|
|
r_core_cmd0 (core, ".ia*");
|
|
}
|
|
#endif
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_ioff(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->io->ff = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_io_oxff(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->io->Oxff = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_filepath(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
char *pikaboo = strstr (node->value, "://");
|
|
if (pikaboo) {
|
|
if (pikaboo[3] == '/') {
|
|
r_config_set (core->config, "file.lastpath", node->value);
|
|
char *ovalue = node->value;
|
|
node->value = strdup (pikaboo + 3);
|
|
free (ovalue);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
r_config_set (core->config, "file.lastpath", node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_ioautofd(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->io->autofd = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scr_color_grep(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
|
|
/* Let cons know we have a new pager. */
|
|
core->cons->grep_color = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_pager(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
|
|
/* Let cons know we have a new pager. */
|
|
core->cons->pager = node->value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_breaklines(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->break_lines = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_fps(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->fps = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrbreakword(void* user, void* data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (*node->value) {
|
|
r_cons_breakword (node->value);
|
|
} else {
|
|
r_cons_breakword (NULL);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrcolumns(void* user, void* data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
RCore *core = (RCore*) user;
|
|
int n = atoi (node->value);
|
|
core->cons->force_columns = n;
|
|
core->dbg->regcols = n / 20;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrfgets(void* user, void* data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->cons->user_fgets = node->i_value
|
|
? NULL : (void *)r_core_fgets;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrhtml(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->is_html = node->i_value;
|
|
// TODO: control error and restore old value (return false?) show errormsg?
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrhighlight(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_highlight (node->value);
|
|
return true;
|
|
}
|
|
|
|
#if __WINDOWS__ && !__CYGWIN__
|
|
static int scr_ansicon(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->ansicon = node->i_value;
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
static int cb_screcho(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->echo = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrlinesleep(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->linesleep = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrpagesize(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->pagesize= node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrflush(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->flush = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrstrconv(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (node->value[0] == '?') {
|
|
if (strlen (node->value) > 1 && node->value[1] == '?') {
|
|
r_cons_printf ("Valid values for scr.strconv:\n"
|
|
" asciiesc convert to ascii with non-ascii chars escaped\n"
|
|
" asciidot convert to ascii with non-ascii chars turned into a dot (except control chars stated below)\n"
|
|
"\n"
|
|
"Ascii chars are in the range 0x20-0x7e. Always escaped control chars are alert (\\a),\n"
|
|
"backspace (\\b), formfeed (\\f), newline (\\n), carriage return (\\r), horizontal tab (\\t)\n"
|
|
"and vertical tab (\\v). Also, double quotes (\\\") are always escaped, but backslashes (\\\\)\n"
|
|
"are only escaped if str.escbslash = true.\n");
|
|
} else {
|
|
print_node_options (node);
|
|
}
|
|
return false;
|
|
} else {
|
|
free ((char *)core->print->strconv_mode);
|
|
core->print->strconv_mode = strdup (node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_graphformat(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (!strcmp (node->value, "?")) {
|
|
r_cons_printf ("png\njpg\npdf\nps\nsvg\njson\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
static int cb_exectrap(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
RCore *core = (RCore*) user;
|
|
if (core->anal && core->anal->esil) {
|
|
core->anal->esil->exectrap = node->i_value;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_iotrap(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
RCore *core = (RCore*) user;
|
|
if (core->anal && core->anal->esil) {
|
|
core->anal->esil->iotrap = node->i_value;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrint(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value && r_sandbox_enable (0)) {
|
|
return false;
|
|
}
|
|
r_cons_singleton ()->is_interactive = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrnkey(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (!strcmp (node->value, "help") || *node->value == '?') {
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrprompt(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_line_singleton()->echo = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_scrrows(void* user, void* data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
int n = atoi (node->value);
|
|
((RCore *)user)->cons->force_rows = n;
|
|
return true;
|
|
}
|
|
|
|
static int cb_contiguous(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->search->contiguous = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_searchalign(void *user, void *data) {
|
|
RCore *core = (RCore *)user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->search->align = node->i_value;
|
|
core->print->addrmod = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_segoff(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->print->flags |= R_PRINT_FLAGS_SEGOFF;
|
|
} else {
|
|
core->print->flags &= (((ut32)-1) & (~R_PRINT_FLAGS_SEGOFF));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_seggrn(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->assembler->seggrn = node->i_value;
|
|
core->anal->seggrn = node->i_value;
|
|
core->print->seggrn = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
|
|
static int cb_stopthreads(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->dbg->stop_all_threads = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_swstep(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->dbg->swstep = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_consbreak(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->dbg->consbreak = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_teefile(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
r_cons_singleton ()->teefile = node->value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_trace(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->dbg->trace->enabled = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_tracetag(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->dbg->trace->tag = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_utf8(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->cons->use_utf8 = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_utf8_curvy(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->cons->use_utf8_curvy = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_zoombyte(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
switch (*node->value) {
|
|
case 'p': case 'f': case 's': case '0':
|
|
case 'F': case 'e': case 'h':
|
|
core->print->zoom->mode = *node->value;
|
|
break;
|
|
default:
|
|
r_cons_printf ("p\nf\ns\n0\nF\ne\nh\n");
|
|
// eprintf ("Invalid zoom.byte value. See pz? for help\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_binverbose(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->bin->verbose = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_rawstr(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->bin->rawstr = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_debase64(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
core->bin->debase64 = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_binstrings(void *user, void *data) {
|
|
const ut32 req = R_BIN_REQ_STRINGS;
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (node->i_value) {
|
|
core->bin->filter_rules |= req;
|
|
} else {
|
|
core->bin->filter_rules &= ~req;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_bindbginfo(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (!core || !core->bin) {
|
|
return false;
|
|
}
|
|
core->bin->want_dbginfo = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_binprefix(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (!core || !core->bin) {
|
|
return false;
|
|
}
|
|
if (node->value && *node->value) {
|
|
if (!strcmp (node->value, "auto")) {
|
|
if (!core->bin->file) {
|
|
return false;
|
|
}
|
|
char *name = (char *)r_file_basename (core->bin->file);
|
|
if (name) {
|
|
r_name_filter (name, strlen (name));
|
|
r_str_filter (name, strlen (name));
|
|
core->bin->prefix = strdup (name);
|
|
free (name);
|
|
}
|
|
} else {
|
|
core->bin->prefix = node->value;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_binmaxstrbuf(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core->bin) {
|
|
int v = node->i_value;
|
|
ut64 old_v = core->bin->maxstrbuf;
|
|
if (v < 1) {
|
|
v = 4; // HACK
|
|
}
|
|
core->bin->maxstrbuf = v;
|
|
if (v>old_v) {
|
|
r_core_bin_refresh_strings (core);
|
|
}
|
|
return true;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_binmaxstr(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core->bin) {
|
|
int v = node->i_value;
|
|
if (v<1) v = 4; // HACK
|
|
core->bin->maxstrlen = v;
|
|
// TODO: Do not refresh if nothing changed (minstrlen ?)
|
|
r_core_bin_refresh_strings (core);
|
|
return true;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_binminstr(void *user, void *data) {
|
|
RCore *core = (RCore *) user;
|
|
RConfigNode *node = (RConfigNode *) data;
|
|
if (core->bin) {
|
|
int v = node->i_value;
|
|
if (v<1) v = 4; // HACK
|
|
core->bin->minstrlen = v;
|
|
// TODO: Do not refresh if nothing changed (minstrlen ?)
|
|
r_core_bin_refresh_strings (core);
|
|
return true;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_searchin(void *user, void *data) {
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
if (node->value[0] == '?') {
|
|
if (strlen (node->value) > 1 && node->value[1] == '?') {
|
|
r_cons_printf ("Valid values for search.in (depends on .from/.to and io.va):\n"
|
|
"raw search in raw io (ignoring bounds)\n"
|
|
"block search in the current block\n"
|
|
"io.map search in current map\n"
|
|
"io.maps search in all maps\n"
|
|
"io.section search in current mapped section\n"
|
|
"io.sections search in all mapped sections\n"
|
|
"io.sections.write search in all writable marked sections\n"
|
|
"io.sections.exec search in all executable marked sections\n"
|
|
"dbg.stack search in the stack\n"
|
|
"dbg.heap search in the heap\n"
|
|
"dbg.map search in current memory map\n"
|
|
"dbg.maps search in all memory maps\n"
|
|
"dbg.maps.exec search in all executable marked memory maps\n"
|
|
"dbg.maps.write search in all writable marked memory maps\n"
|
|
"anal.fcn search in the current function\n"
|
|
"anal.bb search in the current basic-block\n");
|
|
} else {
|
|
print_node_options (node);
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int __dbg_swstep_getter(void *user, RConfigNode *node) {
|
|
RCore *core = (RCore*)user;
|
|
node->i_value = core->dbg->swstep;
|
|
return true;
|
|
}
|
|
|
|
static int cb_dirpfx(RCore *core, RConfigNode *node) {
|
|
r_sys_prefix (node->value);
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_gp(RCore *core, RConfigNode *node) {
|
|
core->anal->gp = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_from(RCore *core, RConfigNode *node) {
|
|
if (r_config_get_i (core->config, "anal.limits")) {
|
|
r_anal_set_limits (core->anal,
|
|
r_config_get_i (core->config, "anal.from"),
|
|
r_config_get_i (core->config, "anal.to"));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_limits(void *user, RConfigNode *node) {
|
|
RCore *core = (RCore*)user;
|
|
if (node->i_value) {
|
|
r_anal_set_limits (core->anal,
|
|
r_config_get_i (core->config, "anal.from"),
|
|
r_config_get_i (core->config, "anal.to"));
|
|
} else {
|
|
r_anal_unset_limits (core->anal);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int cb_anal_rnr(void *user, RConfigNode *node) {
|
|
RCore *core = (RCore*)user;
|
|
core->anal->recursive_noreturn = node->i_value;
|
|
return 1;
|
|
}
|
|
|
|
static int cb_anal_jmptbl(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.jmptbl = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_cjmpref(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.cjmpref = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_jmpref(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.jmpref = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_jmpabove(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.jmpabove = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_followdatarefs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.followdatarefs = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_searchstringrefs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.searchstringrefs = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_pushret(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.pushret = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_brokenrefs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.followbrokenfcnsrefs = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_bbs_alignment(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.bbs_alignment = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_bb_max_size(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->anal->opt.bb_max_size = node->i_value;
|
|
return true;
|
|
}
|
|
|
|
static int cb_anal_cpp_abi(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
|
|
if (*node->value == '?') {
|
|
print_node_options (node);
|
|
return false;
|
|
}
|
|
|
|
if (*node->value) {
|
|
if (strcmp (node->value, "itanium") == 0) {
|
|
core->anal->cpp_abi = R_ANAL_CPP_ABI_ITANIUM;
|
|
return true;
|
|
} else if (strcmp (node->value, "msvc") == 0) {
|
|
core->anal->cpp_abi = R_ANAL_CPP_ABI_MSVC;
|
|
return true;
|
|
}
|
|
eprintf ("anal.cpp.abi: cannot find '%s'\n", node->value);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int cb_linesto(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
ut64 from = (ut64)r_config_get_i (core->config, "lines.from");
|
|
int io_sz = r_io_size (core->io);
|
|
ut64 to = r_num_math (core->num, node->value);
|
|
if (to == 0) {
|
|
core->print->lines_cache_sz = -1; //r_core_lines_initcache (core, from, to);
|
|
return false;
|
|
}
|
|
if (to > from+io_sz) {
|
|
eprintf ("ERROR: \"lines.to\" can't exceed addr 0x%08"PFMT64x
|
|
" 0x%08"PFMT64x" %d\n", from, to, io_sz);
|
|
return true;
|
|
}
|
|
if (to > from) {
|
|
core->print->lines_cache_sz = r_core_lines_initcache (core, from, to);
|
|
//if (core->print->lines_cache_sz == -1) { eprintf ("ERROR: Can't allocate memory\n"); }
|
|
} else {
|
|
eprintf ("Invalid range 0x%08"PFMT64x" .. 0x%08"PFMT64x"\n", from, to);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_linesabs(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
core->print->lines_abs = node->i_value;
|
|
if (core->print->lines_abs && core->print->lines_cache_sz <= 0) {
|
|
ut64 from = (ut64)r_config_get_i (core->config, "lines.from");
|
|
ut64 to = (ut64)r_config_get_i (core->config, "lines.to");
|
|
core->print->lines_cache_sz = r_core_lines_initcache (core, from, to);
|
|
if (core->print->lines_cache_sz == -1) {
|
|
eprintf ("ERROR: \"lines.from\" and \"lines.to\" must be set\n");
|
|
} else {
|
|
eprintf ("Found %d lines\n", core->print->lines_cache_sz-1);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_malloc(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
|
|
if (node->value) {
|
|
if (!strcmp ("jemalloc", node->value) || !strcmp ("glibc", node->value)) {
|
|
if (core->dbg) {
|
|
core->dbg->malloc = data;
|
|
}
|
|
}
|
|
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cb_dbgsnap(void *user, void *data) {
|
|
RCore *core = (RCore*) user;
|
|
RConfigNode *node = (RConfigNode*) data;
|
|
|
|
if (node->value){
|
|
r_debug_session_path (core->dbg, node->value);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static char *getViewerPath() {
|
|
int i;
|
|
const char *viewers[] = {
|
|
"open",
|
|
"geeqie",
|
|
"gqview",
|
|
"eog",
|
|
"xdg-open",
|
|
NULL
|
|
};
|
|
for (i = 0; viewers[i]; i++) {
|
|
char *dotPath = r_file_path (viewers[i]);
|
|
if (dotPath && strcmp (dotPath, viewers[i])) {
|
|
return dotPath;
|
|
}
|
|
free (dotPath);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
R_API char* r_core_graph_cmd(RCore *core, char *r2_cmd) {
|
|
char *cmd = NULL;
|
|
char *xdotPath = r_file_path ("xdot");
|
|
const char *ext = r_config_get (core->config, "graph.extension");
|
|
if (r_file_exists (xdotPath)) {
|
|
cmd = r_str_newf ("%s > a.dot;!xdot a.dot", r2_cmd);
|
|
} else {
|
|
char *dotPath = r_file_path ("dot");
|
|
if (r_file_exists (dotPath)) {
|
|
R_FREE (dotPath);
|
|
char *viewer = getViewerPath();
|
|
if (viewer) {
|
|
cmd = r_str_newf ("%s > a.dot;!dot -T%s -oa.%s a.dot;!%s a.%s", r2_cmd, ext, ext, viewer, ext);
|
|
free (viewer);
|
|
} else {
|
|
cmd = "?e cannot find a valid picture viewer";
|
|
}
|
|
} else {
|
|
cmd = r_str_new ("agf");
|
|
}
|
|
free (dotPath);
|
|
}
|
|
free (xdotPath);
|
|
return cmd;
|
|
}
|
|
|
|
|
|
|
|
#define SLURP_LIMIT (10*1024*1024)
|
|
R_API int r_core_config_init(RCore *core) {
|
|
int i;
|
|
char buf[128], *p, *tmpdir;
|
|
RConfigNode *n;
|
|
RConfig *cfg = core->config = r_config_new (core);
|
|
if (!cfg) {
|
|
return 0;
|
|
}
|
|
cfg->cb_printf = r_cons_printf;
|
|
cfg->num = core->num;
|
|
/* dir.prefix is used in other modules, set it first */
|
|
{
|
|
char *pfx = r_sys_getenv("R2_PREFIX");
|
|
#if __WINDOWS__
|
|
char invoke_dir[MAX_PATH];
|
|
if (!pfx && r_sys_get_src_dir_w32 (invoke_dir)) {
|
|
pfx = strdup (invoke_dir);
|
|
}
|
|
#endif
|
|
if (!pfx) {
|
|
pfx = strdup (R2_PREFIX);
|
|
}
|
|
SETCB ("dir.prefix", pfx, (RConfigCallback)&cb_dirpfx, "Default prefix r2 was compiled for");
|
|
free (pfx);
|
|
}
|
|
/* pdb */
|
|
SETPREF ("pdb.useragent", "Microsoft-Symbol-Server/6.11.0001.402", "User agent for Microsoft symbol server");
|
|
SETPREF ("pdb.server", "https://msdl.microsoft.com/download/symbols", "Base URL for Microsoft symbol server");
|
|
{
|
|
char *pdb_path = r_str_home(R2_HOME_PDB);
|
|
SETPREF("pdb.symstore", pdb_path, "Path to downstream symbol store");
|
|
R_FREE(pdb_path);
|
|
}
|
|
SETI ("pdb.extract", 1, "Avoid extract of the pdb file, just download");
|
|
SETI ("pdb.autoload", false, "Automatically load the required pdb files for loaded DLLs");
|
|
|
|
/* anal */
|
|
SETPREF ("anal.fcnprefix", "fcn", "Prefix new function names with this");
|
|
SETPREF ("anal.a2f", "false", "Use the new WIP analysis algorithm (core/p/a2f), anal.depth ignored atm");
|
|
SETICB ("anal.gp", 0, (RConfigCallback)&cb_anal_gp, "Set the value of the GP register (MIPS)");
|
|
SETI ("anal.gp2", 0, "Set anal.gp before emulating each instruction (workaround)");
|
|
SETCB ("anal.limits", "false", (RConfigCallback)&cb_anal_limits, "Restrict analysis to address range [anal.from:anal.to]");
|
|
SETCB ("anal.rnr", "false", (RConfigCallback)&cb_anal_rnr, "Recursive no return checks (EXPERIMENTAL)");
|
|
SETCB ("anal.limits", "false", (RConfigCallback)&cb_anal_limits, "Restrict analysis to address range [anal.from:anal.to]");
|
|
SETICB ("anal.from", -1, (RConfigCallback)&cb_anal_from, "Lower limit on the address range for analysis");
|
|
SETICB ("anal.to", -1, (RConfigCallback)&cb_anal_from, "Upper limit on the address range for analysis");
|
|
n = NODECB ("anal.in", "io.maps", &cb_searchin);
|
|
SETDESC (n, "Specify search boundaries for analysis");
|
|
SETOPTIONS (n, "raw", "block", "io.map", "io.maps",
|
|
"io.sections", "io.sections.write", "io.sections.exec", "io.sections.readonly",
|
|
"dbg.stack", "dbg.heap", "dbg.map",
|
|
"dbg.maps", "dbg.maps.exec", "dbg.maps.write", "dbg.maps.readonly",
|
|
"anal.fcn", "anal.bb", NULL);
|
|
SETI ("anal.timeout", 0, "Stop analyzing after a couple of seconds");
|
|
|
|
SETCB ("anal.armthumb", "false", &cb_analarmthumb, "aae computes arm/thumb changes (lot of false positives ahead)");
|
|
SETCB ("anal.eobjmp", "false", &cb_analeobjmp, "jmp is end of block mode (option)");
|
|
SETCB ("anal.afterjmp", "true", &cb_analafterjmp, "Continue analysis after jmp/ujmp");
|
|
SETICB ("anal.depth", 64, &cb_analdepth, "Max depth at code analysis"); // XXX: warn if depth is > 50 .. can be problematic
|
|
SETICB ("anal.sleep", 0, &cb_analsleep, "Sleep N usecs every so often during analysis. Avoid 100% CPU usage");
|
|
SETPREF ("anal.calls", "false", "Make basic af analysis walk into calls");
|
|
SETPREF ("anal.autoname", "true", "Automatically set a name for the functions, may result in some false positives");
|
|
SETPREF ("anal.hasnext", "false", "Continue analysis after each function");
|
|
SETPREF ("anal.esil", "false", "Use the new ESIL code analysis");
|
|
SETCB ("anal.strings", "false", &cb_analstrings, "Identify and register strings during analysis (aar only)");
|
|
SETCB ("anal.vars", "true", &cb_analvars, "Analyze local variables and arguments");
|
|
SETPREF ("anal.vinfun", "true", "Search values in functions (aav) (false by default to only find on non-code)");
|
|
SETPREF ("anal.vinfunrange", "false", "Search values outside function ranges (requires anal.vinfun=false)\n");
|
|
SETCB ("anal.nopskip", "true", &cb_analnopskip, "Skip nops at the beginning of functions");
|
|
SETCB ("anal.hpskip", "false", &cb_analhpskip, "Skip `mov reg, reg` and `lea reg, [reg] at the beginning of functions");
|
|
SETCB ("anal.noncode", "false", &cb_analnoncode, "Analyze data as code");
|
|
n = NODECB ("anal.arch", R_SYS_ARCH, &cb_analarch);
|
|
SETDESC (n, "Select the architecture to use");
|
|
update_analarch_options (core, n);
|
|
SETCB ("anal.cpu", R_SYS_ARCH, &cb_analcpu, "Specify the anal.cpu to use");
|
|
SETPREF ("anal.prelude", "", "Specify an hexpair to find preludes in code");
|
|
SETCB ("anal.split", "true", &cb_analsplit, "Split functions into basic blocks in analysis");
|
|
SETCB ("anal.recont", "false", &cb_analrecont, "End block after splitting a basic block instead of error"); // testing
|
|
SETCB ("anal.ijmp", "false", &cb_analijmp, "Follow the indirect jumps in function analysis"); // testing
|
|
SETI ("anal.ptrdepth", 3, "Maximum number of nested pointers to follow in analysis");
|
|
SETICB ("anal.maxreflines", 0, &cb_analmaxrefs, "Maximum number of reflines to be analyzed and displayed in asm.lines with pd");
|
|
|
|
SETCB ("anal.jmptbl", "false", &cb_anal_jmptbl, "Analyze jump tables in switch statements");
|
|
|
|
SETCB ("anal.cjmpref", "false", &cb_anal_cjmpref, "Create references for conditional jumps");
|
|
SETCB ("anal.jmpref", "true", &cb_anal_jmpref, "Create references for unconditional jumps");
|
|
|
|
SETCB ("anal.jmpabove", "true", &cb_anal_jmpabove, "Jump above function pointer");
|
|
SETCB ("anal.datarefs", "false", &cb_anal_followdatarefs, "Follow data references for code coverage");
|
|
SETCB ("anal.brokenrefs", "false", &cb_anal_brokenrefs, "Follow function references as well if function analysis was failed");
|
|
|
|
SETCB ("anal.refstr", "false", &cb_anal_searchstringrefs, "Search string references in data references");
|
|
SETCB ("anal.bb.split", "true", &cb_analbbsplit, "Use the experimental basic block split for JMPs");
|
|
SETCB ("anal.bb.align", "0x10", &cb_anal_bbs_alignment, "Possible space between basic blocks");
|
|
SETCB ("anal.bb.maxsize", "1024", &cb_anal_bb_max_size, "Maximum basic block size");
|
|
SETCB ("anal.pushret", "false", &cb_anal_pushret, "Analyze push+ret as jmp");
|
|
|
|
n = NODECB ("anal.cpp.abi", "itanium", &cb_anal_cpp_abi);
|
|
SETDESC (n, "Select C++ ABI (Compiler)");
|
|
SETOPTIONS (n, "itanium", "msvc", NULL);
|
|
|
|
#if __linux__ && __GNU_LIBRARY__ && __GLIBC__ && __GLIBC_MINOR__
|
|
SETCB("dbg.malloc", "glibc", &cb_malloc, "Choose malloc structure parser");
|
|
#else
|
|
SETCB("dbg.malloc", "jemalloc", &cb_malloc, "Choose malloc structure parser");
|
|
#endif
|
|
|
|
SETPREF ("esil.prestep", "true", "Step before esil evaluation in `de` commands");
|
|
SETPREF ("esil.fillstack", "", "Initialize ESIL stack with (random, debrujn, sequence, zeros, ...)");
|
|
SETICB ("esil.verbose", 0, &cb_esilverbose, "Show ESIL verbose level (0, 1, 2)");
|
|
SETICB ("esil.gotolimit", core->anal->esil_goto_limit, &cb_gotolimit, "Maximum number of gotos per ESIL expression");
|
|
SETICB ("esil.stack.depth", 32, &cb_esilstackdepth, "Number of elements that can be pushed on the esilstack");
|
|
SETI ("esil.stack.size", 0xf0000, "Number of elements that can be pushed on the esilstack");
|
|
SETI ("esil.stack.addr", 0x100000, "Number of elements that can be pushed on the esilstack");
|
|
SETPREF ("esil.stack.pattern", "0", "Specify fill pattern to initialize the stack (0, w, d, i)");
|
|
SETI ("esil.addr.size", 64, "Maximum address size in accessed by the ESIL VM");
|
|
|
|
/* asm */
|
|
//asm.os needs to be first, since other asm.* depend on it
|
|
n = NODECB ("asm.os", R_SYS_OS, &cb_asmos);
|
|
SETDESC (n, "Select operating system (kernel)");
|
|
SETOPTIONS (n, "ios", "dos", "darwin", "linux", "freebsd", "openbsd", "netbsd", "windows", NULL);
|
|
SETI ("asm.xrefs.fold", 5, "Maximum number of xrefs to be displayed as list (use columns above)");
|
|
SETI ("asm.xrefs.max", 20, "Maximum number of xrefs to be displayed without folding");
|
|
SETCB ("asm.invhex", "false", &cb_asm_invhex, "Show invalid instructions as hexadecimal numbers");
|
|
SETPREF ("asm.meta", "true", "Display the code/data/format conversions in disasm");
|
|
SETPREF ("asm.bytes", "true", "Display the bytes of each instruction");
|
|
SETPREF ("asm.midcursor", "false", "Cursor in visual disasm mode breaks the instruction");
|
|
SETPREF ("asm.cmt.flgrefs", "true", "Show comment flags associated to branch reference");
|
|
SETPREF ("asm.cmt.right", "true", "Show comments at right of disassembly if they fit in screen");
|
|
SETI ("asm.cmt.col", 71, "Column to align comments");
|
|
SETICB ("asm.pcalign", 0, &cb_asm_pcalign, "Only recognize as valid instructions aligned to this value");
|
|
// maybe rename to asm.cmt.calls
|
|
SETPREF ("asm.calls", "true", "Show callee function related info as comments in disasm");
|
|
SETPREF ("asm.bbline", "false", "Show empty line after every basic block");
|
|
SETPREF ("asm.comments", "true", "Show comments in disassembly view");
|
|
SETPREF ("asm.jmphints", "true", "Show jump hints [numbers] in disasm");
|
|
SETPREF ("asm.jmpsub", "false", "Always substitute jump, call and branch targets in disassembly");
|
|
SETPREF ("asm.leahints", "false", "Show LEA hints [numbers] in disasm");
|
|
SETPREF ("asm.slow", "true", "Perform slow analysis operations in disasm");
|
|
SETPREF ("asm.decode", "false", "Use code analysis as a disassembler");
|
|
SETICB ("asm.imm.arm", false, &cb_asm_armimm, "Display # for immediates in ARM");
|
|
SETPREF ("asm.imm.str", "false", "Show immediates values as strings");
|
|
SETPREF ("asm.imm.trim", "false", "Remove all offsets and constants from disassembly");
|
|
SETPREF ("asm.indent", "false", "Indent disassembly based on reflines depth");
|
|
SETI ("asm.indentspace", 2, "How many spaces to indent the code");
|
|
SETPREF ("asm.dwarf", "false", "Show dwarf comment at disassembly");
|
|
SETPREF ("asm.dwarf.abspath", "false", "Show absolute path in asm.dwarf");
|
|
SETPREF ("asm.dwarf.file", "true", "Show filename of asm.dwarf in pd");
|
|
SETPREF ("asm.esil", "false", "Show ESIL instead of mnemonic");
|
|
SETPREF ("asm.nodup", "false", "Do not show dupped instructions (collapse disasm)");
|
|
SETPREF ("asm.emu", "false", "Run ESIL emulation analysis on disasm");
|
|
SETPREF ("emu.pre", "false", "Run ESIL emulation starting at the closest flag in pd");
|
|
SETPREF ("emu.stack", "false", "Create a temporary fake stack when emulating in disasm (asm.emu)");
|
|
SETCB ("emu.str", "false", &cb_emustr, "Show only strings if any in the asm.emu output");
|
|
SETPREF ("emu.stroff", "false", "Always show offset when printing asm.emu strings");
|
|
SETPREF ("emu.strinv", "true", "Color-invert asm.emu strings");
|
|
SETPREF ("emu.strflag", "true", "Also show flag (if any) for asm.emu string");
|
|
SETPREF ("emu.write", "false", "Allow asm.emu to modify memory (WARNING)");
|
|
n = NODECB ("emu.skip", "ds", &cb_emuskip);
|
|
SETDESC (n, "Skip metadata of given types in asm.emu");
|
|
SETOPTIONS (n, "d", "c", "s", "f", "m", "h", "C", "r", NULL);
|
|
SETPREF ("asm.filter", "true", "Replace numeric values by flags (e.g. 0x4003e0 -> sym.imp.printf)");
|
|
SETPREF ("asm.strip", "", "strip all instructions given comma separated types");
|
|
SETPREF ("asm.lines.fcn", "true", "Show function boundary lines");
|
|
SETPREF ("asm.flags", "true", "Show flags");
|
|
SETPREF ("asm.flags.offset", "false", "Show offset in flags");
|
|
SETPREF ("asm.flags.inbytes", "false", "Display flags inside the bytes space");
|
|
n = NODEICB ("asm.flags.middle", 2, &cb_midflags);
|
|
SETOPTIONS (n, "0 = do not show flag", "1 = show without realign", "2 = realign at middle flag",
|
|
"3 = realign at middle flag if sym.*", NULL);
|
|
SETDESC (n, "Realign disassembly if there is a flag in the middle of an instruction");
|
|
SETPREF ("asm.lbytes", "true", "Align disasm bytes to left");
|
|
SETPREF ("asm.lines", "true", "Show ASCII-art lines at disassembly");
|
|
SETPREF ("asm.lines.bb", "true", "Show flow lines at jumps");
|
|
SETPREF ("asm.lines.call", "false", "Enable call lines");
|
|
SETPREF ("asm.lines.ret", "false", "Show separator lines after ret");
|
|
SETPREF ("asm.lines.out", "true", "Show out of block lines");
|
|
SETPREF ("asm.lines.right", "false", "Show lines before opcode instead of offset");
|
|
SETPREF ("asm.lines.wide", "false", "Put a space between lines");
|
|
SETICB ("asm.lines.width", 7, &cb_asmlineswidth, "Number of columns for program flow arrows");
|
|
SETICB ("asm.var.submin", 0x100, &cb_asmvarsubmin, "Minimum value to substitute in instructions (asm.var.sub)");
|
|
SETPREF ("asm.middle", "false", "Allow disassembling jumps in the middle of an instruction");
|
|
SETPREF ("asm.noisy", "true", "Show comments considered noisy but possibly useful");
|
|
SETPREF ("asm.offset", "true", "Show offsets at disassembly");
|
|
SETCB ("scr.rainbow", "false", &cb_scrrainbow, "Shows rainbow colors depending of address");
|
|
SETPREF ("asm.reloff", "false", "Show relative offsets instead of absolute address in disasm");
|
|
SETPREF ("asm.reloff.flags", "false", "Show relative offsets to flags (not only functions)");
|
|
SETPREF ("asm.section", "false", "Show section name before offset");
|
|
SETI ("asm.section.col", 20, "Columns width to show asm.section");
|
|
SETCB ("asm.section.sub", "false", &cb_asmsecsub, "Show offsets in disasm prefixed with section/map name");
|
|
SETPREF ("asm.pseudo", "false", "Enable pseudo syntax");
|
|
SETPREF ("asm.size", "false", "Show size of opcodes in disassembly (pd)");
|
|
SETPREF ("asm.stackptr", "false", "Show stack pointer at disassembly");
|
|
SETPREF ("asm.cyclespace", "false", "Indent instructions depending on CPU-cycles");
|
|
SETPREF ("asm.cycles", "false", "Show CPU-cycles taken by instruction at disassembly");
|
|
SETI ("asm.tabs", 0, "Use tabs in disassembly");
|
|
SETPREF ("asm.tabs.once", "false", "Only tabulate the opcode, not the arguments");
|
|
SETI ("asm.tabs.off", 0, "tabulate spaces after the offset");
|
|
SETPREF ("asm.trace", "false", "Show execution traces for each opcode");
|
|
SETPREF ("asm.tracespace", "false", "Indent disassembly with trace.count information");
|
|
SETPREF ("asm.ucase", "false", "Use uppercase syntax at disassembly");
|
|
SETPREF ("asm.capitalize", "false", "Use camelcase at disassembly");
|
|
SETPREF ("asm.var", "true", "Show local function variables in disassembly");
|
|
SETPREF ("asm.var.access", "false", "Show accesses of local variables");
|
|
SETPREF ("asm.var.sub", "true", "Substitute variables in disassembly");
|
|
SETI ("asm.var.summary", 0, "Show variables summary instead of full list in disasm (0, 1, 2)");
|
|
SETPREF ("asm.var.subonly", "true", "Substitute the entire variable expression with the local variable name (e.g. [local10h] instead of [ebp+local10h])");
|
|
SETPREF ("asm.relsub", "true", "Substitute pc relative expressions in disasm");
|
|
SETPREF ("asm.cmt.fold", "false", "Fold comments, toggle with Vz");
|
|
SETPREF ("asm.family", "false", "Show family name in disasm");
|
|
SETPREF ("asm.symbol", "false", "Show symbol+delta instead of absolute offset");
|
|
SETPREF ("asm.anal", "false", "Analyze code and refs while disassembling (see anal.strings)");
|
|
SETI ("asm.shortcut", 1, "Shortcut position (-1, 0, 1)");
|
|
SETI ("asm.symbol.col", 40, "Columns width to show asm.section");
|
|
SETCB ("asm.assembler", "", &cb_asmassembler, "Set the plugin name to use when assembling");
|
|
SETPREF ("asm.minicols", "false", "Only show the instruction in the column disasm");
|
|
RConfigNode *asmcpu = NODECB ("asm.cpu", R_SYS_ARCH, &cb_asmcpu);
|
|
SETDESC (asmcpu, "Set the kind of asm.arch cpu");
|
|
RConfigNode *asmarch = NODECB ("asm.arch", R_SYS_ARCH, &cb_asmarch);
|
|
SETDESC (asmarch, "Set the arch to be used by asm");
|
|
/* we need to have both asm.arch and asm.cpu defined before updating options */
|
|
update_asmarch_options (core, asmarch);
|
|
update_asmcpu_options (core, asmcpu);
|
|
n = NODECB ("asm.features", "", &cb_asmfeatures);
|
|
SETDESC (n, "Specify supported features by the target CPU");
|
|
update_asmfeatures_options (core, n);
|
|
SETCB ("asm.parser", "x86.pseudo", &cb_asmparser, "Set the asm parser to use");
|
|
SETCB ("asm.segoff", "false", &cb_segoff, "Show segmented address in prompt (x86-16)");
|
|
SETCB ("asm.decoff", "false", &cb_decoff, "Show segmented address in prompt (x86-16)");
|
|
SETICB ("asm.seggrn", 4, &cb_seggrn, "Segment granularity in bits (x86-16)");
|
|
n = NODECB ("asm.syntax", "intel", &cb_asmsyntax);
|
|
SETDESC (n, "Select assembly syntax");
|
|
SETOPTIONS (n, "att", "intel", "masm", "jz", "regnum", NULL);
|
|
SETI ("asm.nbytes", 6, "Number of bytes for each opcode at disassembly");
|
|
SETPREF ("asm.bytespace", "false", "Separate hexadecimal bytes with a whitespace");
|
|
#if R_SYS_BITS == R_SYS_BITS_64
|
|
SETICB ("asm.bits", 64, &cb_asmbits, "Word size in bits at assembler");
|
|
#else
|
|
SETICB ("asm.bits", 32, &cb_asmbits, "Word size in bits at assembler");
|
|
#endif
|
|
SETPREF ("asm.functions", "true", "Show functions in disassembly");
|
|
SETPREF ("asm.xrefs", "true", "Show xrefs in disassembly");
|
|
SETPREF ("asm.demangle", "true", "Show demangled symbols in disasm");
|
|
SETPREF ("asm.describe", "false", "Show opcode description");
|
|
SETPREF ("asm.hints", "false", "Show hints for magic numbers in disasm");
|
|
SETPREF ("asm.highlight", "", "Highlight current line");
|
|
SETPREF ("asm.marks", "true", "Show marks before the disassembly");
|
|
SETPREF ("asm.cmt.refs", "false", "Show flag and comments from refs in disasm");
|
|
SETPREF ("asm.cmt.patch", "false", "Show patch comments in disasm");
|
|
SETPREF ("asm.cmt.off", "nodup", "Show offset comment in disasm (true, false, nodup)");
|
|
SETPREF ("asm.payloads", "false", "Show payload bytes in disasm");
|
|
n = NODECB ("asm.strenc", "guess", &cb_asmstrenc);
|
|
SETDESC (n, "Assumed string encoding for disasm");
|
|
SETOPTIONS (n, "latin1", "utf8", "utf16le", "utf32le", "guess", NULL);
|
|
SETCB ("bin.usextr", "true", &cb_usextr, "Use extract plugins when loading files");
|
|
SETCB ("bin.useldr", "true", &cb_useldr, "Use loader plugins when loading files");
|
|
SETCB ("bin.strpurge", "", &cb_strpurge, "Purge strings (e bin.strpurge=? provides more detail)");
|
|
SETPREF ("bin.b64str", "false", "Try to debase64 the strings");
|
|
SETPREF ("bin.libs", "false", "Try to load libraries after loading main binary");
|
|
n = NODECB ("bin.strfilter", "", &cb_strfilter);
|
|
SETDESC (n, "Filter strings");
|
|
SETOPTIONS (n, "a", "8", "p", "e", "u", "i", "U", "f", NULL);
|
|
SETCB ("bin.filter", "true", &cb_binfilter, "Filter symbol names to fix dupped names");
|
|
SETCB ("bin.force", "", &cb_binforce, "Force that rbin plugin");
|
|
SETPREF ("bin.lang", "", "Language for bin.demangle");
|
|
SETPREF ("bin.demangle", "true", "Import demangled symbols from RBin");
|
|
SETCB ("bin.demanglecmd", "false", &cb_bdc, "run xcrun swift-demangle and similar if available (SLOW)");
|
|
|
|
/* bin */
|
|
SETI ("bin.baddr", -1, "Base address of the binary");
|
|
SETI ("bin.laddr", 0, "Base address for loading library ('*.so')");
|
|
SETCB ("bin.dbginfo", "true", &cb_bindbginfo, "Load debug information at startup if available");
|
|
SETPREF ("bin.relocs", "true", "Load relocs information at startup if available");
|
|
SETICB ("bin.minstr", 0, &cb_binminstr, "Minimum string length for r_bin");
|
|
SETICB ("bin.maxstr", 0, &cb_binmaxstr, "Maximum string length for r_bin");
|
|
SETICB ("bin.maxstrbuf", 1024*1024*10, & cb_binmaxstrbuf, "Maximum size of range to load strings from");
|
|
SETCB ("bin.prefix", NULL, &cb_binprefix, "Prefix all symbols/sections/relocs with a specific string");
|
|
SETCB ("bin.rawstr", "false", &cb_rawstr, "Load strings from raw binaries");
|
|
SETCB ("bin.strings", "true", &cb_binstrings, "Load strings from rbin on startup");
|
|
SETCB ("bin.debase64", "false", &cb_debase64, "Try to debase64 all strings");
|
|
SETPREF ("bin.classes", "true", "Load classes from rbin on startup");
|
|
SETCB ("bin.verbose", "true", &cb_binverbose, "Show RBin warnings when loading binaries");
|
|
|
|
/* prj */
|
|
SETPREF ("prj.name", "", "Name of current project");
|
|
SETPREF ("prj.files", "false", "Save the target binary inside the project directory");
|
|
SETPREF ("prj.git", "false", "Every project is a git repo and saving is committing");
|
|
SETPREF ("prj.zip", "false", "Use ZIP format for project files");
|
|
SETPREF ("prj.gpg", "false", "TODO: Encrypt project with GnuPGv2");
|
|
SETPREF ("prj.simple", "false", "Use simple project saving style (functions, comments, options)");
|
|
|
|
/* cfg */
|
|
SETPREF ("cfg.plugins", "true", "Load plugins at startup");
|
|
SETCB ("time.fmt", "%Y-%m-%d %H:%M:%S %z", &cb_cfgdatefmt, "Date format (%Y-%m-%d %H:%M:%S %z)");
|
|
SETICB ("time.zone", 0, &cb_timezone, "Time zone, in hours relative to GMT: +2, -1,..");
|
|
SETCB ("cfg.log", "false", &cb_cfglog, "Log changes using the T api needed for realtime syncing");
|
|
SETPREF ("cfg.newtab", "false", "Show descriptions in command completion");
|
|
SETCB ("cfg.debug", "false", &cb_cfgdebug, "Debugger mode");
|
|
p = r_sys_getenv ("EDITOR");
|
|
#if __WINDOWS__ && !__CYGWIN__
|
|
r_config_set (cfg, "cfg.editor", p? p: "notepad");
|
|
#else
|
|
r_config_set (cfg, "cfg.editor", p? p: "vi");
|
|
#endif
|
|
free (p);
|
|
r_config_desc (cfg, "cfg.editor", "Select default editor program");
|
|
SETPREF ("cfg.user", r_sys_whoami (buf), "Set current username/pid");
|
|
SETCB ("cfg.fortunes", "true", &cb_cfg_fortunes, "If enabled show tips at start");
|
|
SETCB ("cfg.fortunes.type", "tips,fun", &cb_cfg_fortunes_type, "Type of fortunes to show (tips, fun, nsfw, creepy)");
|
|
SETPREF ("cfg.fortunes.clippy", "false", "Use ?E instead of ?e");
|
|
SETPREF ("cfg.fortunes.tts", "false", "Speak out the fortune");
|
|
SETI ("cfg.hashlimit", SLURP_LIMIT, "If the file is bigger than hashlimit, do not compute hashes");
|
|
SETPREF ("cfg.prefixdump", "dump", "Filename prefix for automated dumps");
|
|
SETCB ("cfg.sandbox", "false", &cb_cfgsanbox, "Sandbox mode disables systems and open on upper directories");
|
|
SETPREF ("cfg.wseek", "false", "Seek after write");
|
|
SETCB ("cfg.bigendian", "false", &cb_bigendian, "Use little (false) or big (true) endianness");
|
|
|
|
// zign
|
|
SETPREF ("zign.prefix", "sign", "Default prefix for zignatures matches");
|
|
SETI ("zign.maxsz", 500, "Maximum zignature length");
|
|
SETI ("zign.minsz", 16, "Minimum zignature length for matching");
|
|
SETI ("zign.mincc", 10, "Minimum cyclomatic complexity for matching");
|
|
SETPREF ("zign.graph", "true", "Use graph metrics for matching");
|
|
SETPREF ("zign.bytes", "true", "Use bytes patterns for matching");
|
|
SETPREF ("zign.offset", "true", "Use original offset for matching");
|
|
SETPREF ("zign.refs", "true", "Use references for matching");
|
|
SETPREF ("zign.autoload", "false", "Autoload all zignatures located in " R_JOIN_2_PATHS ("~", R2_HOME_ZIGNS));
|
|
|
|
/* diff */
|
|
SETCB ("diff.sort", "addr", &cb_diff_sort, "Specify function diff sorting column see (e diff.sort=?)");
|
|
SETI ("diff.from", 0, "Set source diffing address for px (uses cc command)");
|
|
SETI ("diff.to", 0, "Set destination diffing address for px (uses cc command)");
|
|
SETPREF ("diff.bare", "false", "Never show function names in diff output");
|
|
SETPREF ("diff.levenstein", "false", "Use faster (and buggy) levenstein algorithm for buffer distance diffing");
|
|
|
|
/* dir */
|
|
SETCB ("dir.dbgsnap", ".", &cb_dbgsnap, "Path to session dump files");
|
|
{
|
|
char *path = r_str_newf (R_JOIN_2_PATHS ("%s", R2_SDB_MAGIC), r_config_get (core->config, "dir.prefix"));
|
|
SETPREF ("dir.magic", path, "Path to r_magic files");
|
|
free (path);
|
|
path = r_str_newf (R_JOIN_2_PATHS ("%s", R2_PLUGINS), r_config_get (core->config, "dir.prefix"));
|
|
SETPREF ("dir.plugins", path, "Path to plugin files to be loaded at startup");
|
|
free (path);
|
|
}
|
|
SETCB ("dir.source", "", &cb_dirsrc, "Path to find source files");
|
|
SETPREF ("dir.types", "/usr/include", "Default path to look for cparse type files");
|
|
#if __ANDROID__
|
|
SETPREF ("dir.projects", "/data/data/org.radare.radare2installer/radare2/projects", "Default path for projects");
|
|
#else
|
|
SETPREF ("dir.projects", R_JOIN_2_PATHS ("~", R2_HOME_PROJECTS), "Default path for projects");
|
|
#endif
|
|
SETCB ("dir.zigns", R_JOIN_2_PATHS ("~", R2_HOME_ZIGNS), &cb_dirzigns, "Default path for zignatures (see zo command)");
|
|
SETPREF ("stack.bytes", "true", "Show bytes instead of words in stack");
|
|
SETPREF ("stack.anotated", "false", "Show anotated hexdump in visual debug");
|
|
SETI ("stack.size", 64, "Size in bytes of stack hexdump in visual debug");
|
|
SETI ("stack.delta", 0, "Delta for the stack dump");
|
|
|
|
SETCB ("dbg.libs", "", &cb_dbg_libs, "If set stop when loading matching libname");
|
|
SETI ("dbg.hwbp", 0, "Set HW or SW breakpoints");
|
|
SETCB ("dbg.unlibs", "", &cb_dbg_unlibs, "If set stop when unloading matching libname");
|
|
SETPREF ("dbg.slow", "false", "Show stack and regs in visual mode in a slow but verbose mode");
|
|
SETPREF ("dbg.funcarg", "false", "Display arguments to function call in visual mode");
|
|
|
|
SETPREF ("dbg.bpinmaps", "true", "Force breakpoints to be inside a valid map");
|
|
SETCB ("dbg.forks", "false", &cb_dbg_forks, "Stop execution if fork() is done (see dbg.threads)");
|
|
n = NODECB ("dbg.btalgo", "fuzzy", &cb_dbg_btalgo);
|
|
SETDESC (n, "Select backtrace algorithm");
|
|
SETOPTIONS (n, "default", "fuzzy", "anal", "trace", NULL);
|
|
SETCB ("dbg.threads", "false", &cb_stopthreads, "Stop all threads when debugger breaks (see dbg.forks)");
|
|
SETCB ("dbg.clone", "false", &cb_dbg_clone, "Stop execution if new thread is created");
|
|
SETCB ("dbg.aftersyscall", "true", &cb_dbg_aftersc, "Stop execution before the syscall is executed (see dcs)");
|
|
SETCB ("dbg.execs", "false", &cb_dbg_execs, "Stop execution if new thread is created");
|
|
SETCB ("dbg.profile", "", &cb_runprofile, "Path to RRunProfile file");
|
|
SETCB ("dbg.args", "", &cb_dbg_args, "Set the args of the program to debug");
|
|
SETCB ("dbg.follow.child", "false", &cb_dbg_follow_child, "Continue tracing the child process on fork. By default the parent process is traced");
|
|
/* debug */
|
|
SETCB ("dbg.status", "false", &cb_dbgstatus, "Set cmd.prompt to '.dr*' or '.dr*;drd;sr PC;pi 1;s-'");
|
|
#if DEBUGGER
|
|
SETCB ("dbg.backend", "native", &cb_dbgbackend, "Select the debugger backend");
|
|
#else
|
|
SETCB ("dbg.backend", "esil", &cb_dbgbackend, "Select the debugger backend");
|
|
#endif
|
|
n = NODECB ("dbg.bep", "loader", &cb_dbgbep);
|
|
SETDESC (n, "Break on entrypoint");
|
|
SETOPTIONS (n, "loader", "entry", "constructor", "main", NULL);
|
|
if (core->cons->rows > 30) { // HACKY
|
|
r_config_set_i (cfg, "dbg.follow", 64);
|
|
} else {
|
|
r_config_set_i (cfg, "dbg.follow", 32);
|
|
}
|
|
r_config_desc (cfg, "dbg.follow", "Follow program counter when pc > core->offset + dbg.follow");
|
|
SETCB ("dbg.swstep", "false", &cb_swstep, "Force use of software steps (code analysis+breakpoint)");
|
|
SETPREF ("dbg.trace.inrange", "false", "While tracing, avoid following calls outside specified range");
|
|
SETPREF ("dbg.trace.libs", "true", "Trace library code too");
|
|
SETPREF ("dbg.exitkills", "true", "Kill process on exit");
|
|
SETPREF ("dbg.exe.path", NULL, "Path to binary being debugged");
|
|
SETICB ("dbg.gdb.page_size", 4096, &cb_dbg_gdb_page_size, "Page size on gdb target (useful for QEMU)");
|
|
SETICB ("dbg.gdb.retries", 10, &cb_dbg_gdb_retries, "Number of retries before gdb packet read times out");
|
|
SETCB ("dbg.consbreak", "false", &cb_consbreak, "SIGINT handle for attached processes");
|
|
|
|
r_config_set_getter (cfg, "dbg.swstep", (RConfigCallback)__dbg_swstep_getter);
|
|
|
|
// TODO: This should be specified at first by the debug backend when attaching
|
|
#if __arm__ || __mips__
|
|
SETICB ("dbg.bpsize", 4, &cb_dbgbpsize, "Size of software breakpoints");
|
|
#else
|
|
SETICB ("dbg.bpsize", 1, &cb_dbgbpsize, "Size of software breakpoints");
|
|
#endif
|
|
SETPREF ("dbg.bpsysign", "false", "Ignore system breakpoints");
|
|
SETICB ("dbg.btdepth", 128, &cb_dbgbtdepth, "Depth of backtrace");
|
|
SETCB ("dbg.trace", "false", &cb_trace, "Trace program execution (see asm.trace)");
|
|
SETICB ("dbg.trace.tag", 0, &cb_tracetag, "Trace tag");
|
|
|
|
|
|
/* cmd */
|
|
SETPREF ("cmd.xterm", "xterm -bg black -fg gray -e", "xterm command to spawn with V@");
|
|
SETICB ("cmd.depth", 10, &cb_cmddepth, "Maximum command depth");
|
|
SETPREF ("cmd.bp", "", "Run when a breakpoint is hit");
|
|
SETICB ("cmd.hitinfo", 1, &cb_debug_hitinfo, "Show info when a tracepoint/breakpoint is hit");
|
|
SETPREF ("cmd.times", "", "Run when a command is repeated (number prefix)");
|
|
SETPREF ("cmd.stack", "", "Command to display the stack in visual debug mode");
|
|
SETPREF ("cmd.cprompt", "", "Column visual prompt commands");
|
|
SETPREF ("cmd.gprompt", "", "Graph visual prompt commands");
|
|
SETPREF ("cmd.hit", "", "Run when a search hit is found");
|
|
SETPREF ("cmd.open", "", "Run when file is opened");
|
|
SETCB ("cmd.pdc", "", &cb_cmdpdc, "Select pseudo-decompiler command to run after pdc");
|
|
SETCB ("cmd.log", "", &cb_cmdlog, "Every time a new T log is added run this command");
|
|
SETPREF ("cmd.prompt", "", "Prompt commands");
|
|
SETCB ("cmd.repeat", "false", &cb_cmdrepeat, "Empty command an alias for '..' (repeat last command)");
|
|
SETPREF ("cmd.fcn.new", "", "Run when new function is analyzed");
|
|
SETPREF ("cmd.fcn.delete", "", "Run when a function is deleted");
|
|
SETPREF ("cmd.fcn.rename", "", "Run when a function is renamed");
|
|
SETPREF ("cmd.visual", "", "Replace current print mode");
|
|
SETPREF ("cmd.vprompt", "", "Visual prompt commands");
|
|
|
|
SETCB ("cmd.esil.mdev", "", &cb_cmd_esil_mdev, "Command to run when memory device address is accessed");
|
|
SETCB ("cmd.esil.intr", "", &cb_cmd_esil_intr, "Command to run when an esil interrupt happens");
|
|
SETCB ("cmd.esil.trap", "", &cb_cmd_esil_trap, "Command to run when an esil trap happens");
|
|
SETCB ("cmd.esil.todo", "", &cb_cmd_esil_todo, "Command to run when the esil instruction contains TODO");
|
|
SETCB ("cmd.esil.ioer", "", &cb_cmd_esil_ioer, "Command to run when esil fails to IO (invalid read/write)");
|
|
|
|
/* filesystem */
|
|
n = NODECB ("fs.view", "normal", &cb_fsview);
|
|
SETDESC (n, "Set visibility options for filesystems");
|
|
SETOPTIONS (n, "all", "deleted", "special", NULL);
|
|
|
|
/* hexdump */
|
|
SETPREF ("hex.header", "true", "Show header in hexdumps");
|
|
SETCB ("hex.pairs", "true", &cb_hexpairs, "Show bytes paired in 'px' hexdump");
|
|
SETCB ("hex.compact", "false", &cb_hexcompact, "Show smallest 16 byte col hexdump (60 columns)");
|
|
SETI ("hex.flagsz", 0, "If non zero, overrides the flag size in pxa");
|
|
SETICB ("hex.cols", 16, &cb_hexcols, "Number of columns in hexdump");
|
|
SETI ("hex.pcols", 40, "Number of pixel columns for prc");
|
|
SETI ("hex.depth", 5, "Maximal level of recurrence while telescoping memory");
|
|
SETPREF ("hex.onechar", "false", "Number of columns in hexdump");
|
|
SETICB ("hex.stride", 0, &cb_hexstride, "Line stride in hexdump (default is 0)");
|
|
SETCB ("hex.comments", "true", &cb_hexcomments, "Show comments in 'px' hexdump");
|
|
|
|
/* http */
|
|
SETPREF ("http.log", "true", "Show HTTP requests processed");
|
|
SETPREF ("http.logfile", "", "Specify a log file instead of stderr for http requests");
|
|
SETPREF ("http.cors", "false", "Enable CORS");
|
|
SETPREF ("http.referer", "", "CSFR protection if set");
|
|
SETPREF ("http.dirlist", "false", "Enable directory listing");
|
|
SETPREF ("http.allow", "", "Only accept clients from the comma separated IP list");
|
|
#if __WINDOWS__
|
|
r_config_set (cfg, "http.browser", "start");
|
|
#else
|
|
if (r_file_exists ("/usr/bin/openURL")) { // iOS ericautils
|
|
r_config_set (cfg, "http.browser", "/usr/bin/openURL");
|
|
} else if (r_file_exists ("/system/bin/toolbox")) {
|
|
r_config_set (cfg, "http.browser",
|
|
"LD_LIBRARY_PATH=/system/lib am start -a android.intent.action.VIEW -d");
|
|
} else if (r_file_exists ("/usr/bin/xdg-open")) {
|
|
r_config_set (cfg, "http.browser", "xdg-open");
|
|
} else if (r_file_exists ("/usr/bin/open")) {
|
|
r_config_set (cfg, "http.browser", "open");
|
|
} else {
|
|
r_config_set (cfg, "http.browser", "firefox");
|
|
}
|
|
r_config_desc (cfg, "http.browser", "Command to open HTTP URLs");
|
|
#endif
|
|
SETI ("http.maxsize", 0, "Maximum file size for upload");
|
|
SETPREF ("http.bind", "localhost", "Server address");
|
|
SETPREF ("http.homeroot", R_JOIN_2_PATHS ("~", R2_HOME_WWWROOT), "http home root directory");
|
|
#if __ANDROID__
|
|
SETPREF ("http.root", "/data/data/org.radare.radare2installer/www", "http root directory");
|
|
#else
|
|
SETPREF ("http.root", R2_WWWROOT, "http root directory");
|
|
#endif
|
|
SETPREF ("http.port", "9090", "HTTP server port");
|
|
SETPREF ("http.maxport", "9999", "Last HTTP server port");
|
|
SETPREF ("http.ui", "m", "Default webui (enyo, m, p, t)");
|
|
SETPREF ("http.sandbox", "true", "Sandbox the HTTP server");
|
|
SETI ("http.timeout", 3, "Disconnect clients after N seconds of inactivity");
|
|
SETI ("http.dietime", 0, "Kill server after N seconds with no client");
|
|
SETPREF ("http.verbose", "true", "Output server logs to stdout");
|
|
SETPREF ("http.upget", "false", "/up/ answers GET requests, in addition to POST");
|
|
SETPREF ("http.upload", "false", "Enable file uploads to /up/<filename>");
|
|
SETPREF ("http.uri", "", "Address of HTTP proxy");
|
|
tmpdir = r_file_tmpdir ();
|
|
r_config_set (cfg, "http.uproot", tmpdir);
|
|
free (tmpdir);
|
|
r_config_desc (cfg, "http.uproot", "Path where files are uploaded");
|
|
|
|
/* tcp */
|
|
SETPREF ("tcp.islocal", "false", "Bind a loopback for tcp command server");
|
|
|
|
/* graph */
|
|
SETPREF ("graph.comments", "true", "Show disasm comments in graph");
|
|
SETPREF ("graph.cmtright", "false", "Show comments at right");
|
|
SETCB ("graph.extension", "gif", &cb_graphformat, "Graph extension when using 'w' format (png, jpg, pdf, ps, svg, json)");
|
|
SETPREF ("graph.refs", "false", "Graph references in callgraphs (.agc*;aggi)");
|
|
SETI ("graph.edges", 2, "0=no edges, 1=simple edges, 2=avoid collisions");
|
|
SETI ("graph.layout", 0, "Graph layout (0=vertical, 1=horizontal)");
|
|
SETI ("graph.linemode", 1, "Graph edges (0=diagonal, 1=square)");
|
|
SETPREF ("graph.font", "Courier", "Font for dot graphs");
|
|
SETPREF ("graph.offset", "false", "Show offsets in graphs");
|
|
SETPREF ("graph.web", "false", "Display graph in web browser (VV)");
|
|
SETI ("graph.from", UT64_MAX, "Lower bound address when drawing global graphs");
|
|
SETI ("graph.to", UT64_MAX, "Upper bound address when drawing global graphs");
|
|
SETI ("graph.scroll", 5, "Scroll speed in ascii-art graph");
|
|
SETPREF ("graph.invscroll", "false", "Invert scroll direction in ascii-art graph");
|
|
SETPREF ("graph.title", "", "Title of the graph");
|
|
SETPREF ("graph.gv.node", "", "Graphviz node style. (color=gray, style=filled shape=box)");
|
|
SETPREF ("graph.gv.edge", "", "Graphviz edge style. (arrowhead=\"vee\")");
|
|
SETPREF ("graph.gv.spline", "", "Graphviz spline style. (splines=\"ortho\")");
|
|
SETPREF ("graph.gv.graph", "", "Graphviz global style attributes. (bgcolor=white)");
|
|
SETPREF ("graph.gv.current", "false", "Highlight the current node in graphviz graph.");
|
|
SETPREF ("graph.nodejmps", "true", "Enables shortcuts for every node.");
|
|
|
|
/* hud */
|
|
SETPREF ("hud.path", "", "Set a custom path for the HUD file");
|
|
|
|
SETCB ("esil.exectrap", "false", &cb_exectrap, "trap when executing code in non-executable memory");
|
|
SETCB ("esil.iotrap", "true", &cb_iotrap, "invalid read or writes produce a trap exception");
|
|
SETPREF ("esil.romem", "false", "Set memory as read-only for ESIL");
|
|
SETPREF ("esil.stats", "false", "Statistics from ESIL emulation stored in sdb");
|
|
SETPREF ("esil.nonull", "false", "Prevent memory read, memory write at null pointer");
|
|
SETCB ("esil.mdev.range", "", &cb_mdevrange, "Specify a range of memory to be handled by cmd.esil.mdev");
|
|
|
|
/* scr */
|
|
#if __EMSCRIPTEN__
|
|
r_config_set_cb (cfg, "scr.fgets", "true", cb_scrfgets);
|
|
#else
|
|
r_config_set_cb (cfg, "scr.fgets", "false", cb_scrfgets);
|
|
#endif
|
|
r_config_desc (cfg, "scr.fgets", "Use fgets() instead of dietline for prompt input");
|
|
SETCB ("scr.echo", "false", &cb_screcho, "Show rcons output in realtime to stderr and buffer");
|
|
SETICB ("scr.linesleep", 0, &cb_scrlinesleep, "Flush sleeping some ms in every line");
|
|
SETICB ("scr.pagesize", 1, &cb_scrpagesize, "Flush in pages when scr.linesleep is != 0");
|
|
SETCB ("scr.flush", "false", &cb_scrflush, "Force flush to console in realtime (breaks scripting)");
|
|
/* TODO: rename to asm.color.ops ? */
|
|
SETPREF ("scr.zoneflags", "true", "Show zoneflags in visual mode before the title (see fz?)");
|
|
SETPREF ("scr.slow", "true", "Do slow stuff on visual mode like RFlag.get_at(true)");
|
|
SETPREF ("scr.color.ops", "true", "Colorize numbers and registers in opcodes");
|
|
SETPREF ("scr.color.bytes", "true", "Colorize bytes that represent the opcodes of the instruction");
|
|
#if __WINDOWS__ && !__CYGWIN__
|
|
SETCB ("scr.ansicon", r_str_bool (r_cons_singleton ()->ansicon),
|
|
&scr_ansicon, "Use ANSICON mode or not on Windows");
|
|
#endif
|
|
#if __ANDROID__
|
|
SETPREF ("scr.responsive", "true", "Auto-adjust Visual depending on screen (e.g. unset asm.bytes)");
|
|
#else
|
|
SETPREF ("scr.responsive", "false", "Auto-adjust Visual depending on screen (e.g. unset asm.bytes)");
|
|
#endif
|
|
SETPREF ("scr.wheel.nkey", "false", "Use sn/sp and scr.nkey on wheel instead of scroll");
|
|
SETPREF ("scr.wheel", "true", "Mouse wheel in Visual; temporaryly disable/reenable by right click/Enter)");
|
|
SETPREF ("scr.atport", "false", "V@ starts a background http server and spawns an r2 -C");
|
|
SETI ("scr.wheel.speed", 4, "Mouse wheel speed");
|
|
// DEPRECATED: USES hex.cols now SETI ("scr.colpos", 80, "Column position of cmd.cprompt in visual");
|
|
SETCB ("scr.breakword", "", &cb_scrbreakword, "Emulate console break (^C) when a word is printed (useful for pD)");
|
|
SETCB ("scr.breaklines", "false", &cb_breaklines, "Break lines in Visual instead of truncating them");
|
|
SETICB ("scr.columns", 0, &cb_scrcolumns, "Force console column count (width)");
|
|
SETCB ("scr.rows", "0", &cb_scrrows, "Force console row count (height) ");
|
|
SETICB ("scr.rows", 0, &cb_rows, "Force console row count (height) (duplicate?)");
|
|
SETCB ("scr.fps", "false", &cb_fps, "Show FPS in Visual");
|
|
SETICB ("scr.fix.rows", 0, &cb_fixrows, "Workaround for Linux TTY");
|
|
SETICB ("scr.fix.columns", 0, &cb_fixcolumns, "Workaround for Prompt iOS SSH client");
|
|
SETCB ("scr.highlight", "", &cb_scrhighlight, "Highlight that word at RCons level");
|
|
SETCB ("scr.interactive", "true", &cb_scrint, "Start in interactive mode");
|
|
SETI ("scr.feedback", 1, "Set visual feedback level (1=arrow on jump, 2=every key (useful for videos))");
|
|
SETCB ("scr.html", "false", &cb_scrhtml, "Disassembly uses HTML syntax");
|
|
n = NODECB ("scr.nkey", "flag", &cb_scrnkey);
|
|
SETDESC (n, "Select visual seek mode (affects n/N visual commands)");
|
|
SETOPTIONS (n, "fun", "hit", "flag", NULL);
|
|
SETCB ("scr.pager", "", &cb_pager, "Select pager program (when output overflows the window)");
|
|
SETPREF ("scr.randpal", "false", "Random color palete or just get the next one from 'eco'");
|
|
SETCB ("scr.color.grep", "false", &cb_scr_color_grep, "Enable colors when using ~grep");
|
|
SETPREF ("scr.pipecolor", "false", "Enable colors when using pipes");
|
|
SETPREF ("scr.prompt.file", "false", "Show user prompt file (used by r2 -q)");
|
|
SETPREF ("scr.prompt.flag", "false", "Show flag name in the prompt");
|
|
SETPREF ("scr.prompt.sect", "false", "Show section name in the prompt");
|
|
SETPREF ("scr.tts", "false", "Use tts if available by a command (see ic)");
|
|
SETCB ("scr.prompt", "true", &cb_scrprompt, "Show user prompt (used by r2 -q)");
|
|
SETCB ("scr.tee", "", &cb_teefile, "Pipe output to file of this name");
|
|
SETPREF ("scr.seek", "", "Seek to the specified address on startup");
|
|
SETICB ("scr.color", (core->print->flags&R_PRINT_FLAGS_COLOR)?COLOR_MODE_16:COLOR_MODE_DISABLED, &cb_color, "Enable colors (0: none, 1: ansi, 2: 256 colors, 3: truecolor)");
|
|
SETCB ("scr.null", "false", &cb_scrnull, "Show no output");
|
|
SETCB ("scr.utf8", r_cons_is_utf8()?"true":"false",
|
|
&cb_utf8, "Show UTF-8 characters instead of ANSI");
|
|
SETCB ("scr.utf8.curvy", "false", &cb_utf8_curvy, "Show curved UTF-8 corners (requires scr.utf8)");
|
|
SETPREF ("scr.histsave", "true", "Always save history on exit");
|
|
n = NODECB ("scr.strconv", "asciiesc", &cb_scrstrconv);
|
|
SETDESC (n, "Convert string before display");
|
|
SETOPTIONS (n, "asciiesc", "asciidot", NULL);
|
|
|
|
/* str */
|
|
SETCB ("str.escbslash", "false", &cb_str_escbslash, "Escape the backslash");
|
|
|
|
/* search */
|
|
SETCB ("search.contiguous", "true", &cb_contiguous, "Accept contiguous/adjacent search hits");
|
|
SETICB ("search.align", 0, &cb_searchalign, "Only catch aligned search hits");
|
|
SETI ("search.chunk", 0, "Chunk size for /+ (default size is asm.bits/8");
|
|
SETI ("search.esilcombo", 8, "Stop search after N consecutive hits");
|
|
SETI ("search.distance", 0, "Search string distance");
|
|
SETPREF ("search.flags", "true", "All search results are flagged, otherwise only printed");
|
|
SETPREF ("search.overlap", "false", "Look for overlapped search hits");
|
|
SETI ("search.maxhits", 0, "Maximum number of hits (0: no limit)");
|
|
SETI ("search.from", -1, "Search start address");
|
|
n = NODECB ("search.in", "io.maps", &cb_searchin);
|
|
SETDESC (n, "Specify search boundaries");
|
|
SETOPTIONS (n, "raw", "block", "io.map", "io.maps",
|
|
"io.sections", "io.sections.write", "io.sections.exec", "io.sections.readonly",
|
|
"dbg.stack", "dbg.heap", "dbg.map",
|
|
"dbg.maps", "dbg.maps.exec", "dbg.maps.write", "dbg.maps.readonly",
|
|
"anal.fcn", "anal.bb", NULL);
|
|
SETICB ("search.kwidx", 0, &cb_search_kwidx, "Store last search index count");
|
|
SETPREF ("search.prefix", "hit", "Prefix name in search hits label");
|
|
SETPREF ("search.show", "true", "Show search results");
|
|
SETI ("search.to", -1, "Search end address");
|
|
|
|
/* rop */
|
|
SETI ("rop.len", 5, "Maximum ROP gadget length");
|
|
SETPREF ("rop.sdb", "false", "Cache results in sdb (experimental)");
|
|
SETPREF ("rop.db", "true", "Categorize rop gadgets in sdb");
|
|
SETPREF ("rop.subchains", "false", "Display every length gadget from rop.len=X to 2 in /Rl");
|
|
SETPREF ("rop.conditional", "false", "Include conditional jump, calls and returns in ropsearch");
|
|
SETPREF ("rop.nx", "false", "Include NX/XN/XD sections in ropsearch");
|
|
SETPREF ("rop.comments", "false", "Display comments in rop search output");
|
|
|
|
/* io */
|
|
SETCB ("io.buffer", "false", &cb_iobuffer, "Load and use buffer cache if enabled");
|
|
SETI ("io.buffer.from", 0, "Lower address of buffered cache");
|
|
SETI ("io.buffer.to", 0, "Higher address of buffered cache");
|
|
SETCB ("io.cache", "false", &cb_io_cache, "Change both of io.cache.{read,write}");
|
|
SETCB ("io.cache.auto", "false", &cb_io_cache_mode, "Automatic cache all reads in the IO backend");
|
|
SETCB ("io.cache.read", "false", &cb_io_cache_read, "Enable read cache for vaddr (or paddr when io.va=0)");
|
|
SETCB ("io.cache.write", "false", &cb_io_cache_write, "Enable write cache for vaddr (or paddr when io.va=0)");
|
|
SETCB ("io.pcache", "false", &cb_iopcache, "io.cache for p-level");
|
|
SETCB ("io.pcache.write", "false", &cb_iopcachewrite, "Enable write-cache");
|
|
SETCB ("io.pcache.read", "false", &cb_iopcacheread, "Enable read-cache");
|
|
SETCB ("io.ff", "true", &cb_ioff, "Fill invalid buffers with 0xff instead of returning error");
|
|
SETPREF("io.exec", "true", "See !!r2 -h~-x");
|
|
SETICB ("io.0xff", 0xff, &cb_io_oxff, "Use this value instead of 0xff to fill unallocated areas");
|
|
SETCB ("io.aslr", "false", &cb_ioaslr, "Disable ASLR for spawn and such");
|
|
SETCB ("io.va", "true", &cb_iova, "Use virtual address layout");
|
|
SETCB ("io.autofd", "true", &cb_ioautofd, "Change fd when opening a new file");
|
|
|
|
/* file */
|
|
SETPREF ("file.desc", "", "User defined file description (used by projects)");
|
|
SETPREF ("file.md5", "", "MD5 sum of current file");
|
|
SETPREF ("file.info", "true", "RBin info loaded");
|
|
SETPREF ("file.offset", "", "Offset where the file will be mapped at");
|
|
SETCB ("file.path", "", &cb_filepath, "Path of current file");
|
|
SETPREF ("file.lastpath", "", "Path of current file");
|
|
SETPREF ("file.sha1", "", "SHA1 hash of current file");
|
|
SETPREF ("file.type", "", "Type of current file");
|
|
SETI ("file.loadalign", 1024, "Alignment of load addresses");
|
|
SETI ("file.openmany", 1, "Maximum number of files opened at once");
|
|
SETPREF ("file.nowarn", "true", "Suppress file loading warning messages");
|
|
SETPREF ("file.location", "", "Is the file 'local', 'remote', or 'memory'");
|
|
/* magic */
|
|
SETI ("magic.depth", 100, "Recursivity depth in magic description strings");
|
|
|
|
/* rap */
|
|
SETPREF ("rap.loop", "true", "Run rap as a forever-listening daemon");
|
|
|
|
/* nkeys */
|
|
SETPREF ("key.s", "", "override step into action");
|
|
SETPREF ("key.S", "", "override step over action");
|
|
for (i = 1; i < 13; i++) {
|
|
snprintf (buf, sizeof (buf), "key.f%d", i);
|
|
snprintf (buf + 10, sizeof (buf) - 10,
|
|
"Run this when F%d key is pressed in visual mode", i);
|
|
switch (i) {
|
|
default: p = ""; break;
|
|
}
|
|
r_config_set (cfg, buf, p);
|
|
r_config_desc (cfg, buf, buf+10);
|
|
}
|
|
|
|
/* zoom */
|
|
SETCB ("zoom.byte", "h", &cb_zoombyte, "Zoom callback to calculate each byte (See pz? for help)");
|
|
SETI ("zoom.from", 0, "Zoom start address");
|
|
SETI ("zoom.maxsz", 512, "Zoom max size of block");
|
|
SETI ("zoom.to", 0, "Zoom end address");
|
|
n = NODECB ("zoom.in", "io.map", &cb_searchin);
|
|
SETDESC (n, "Specify boundaries for zoom");
|
|
SETOPTIONS (n, "raw", "block", "io.map", "io.maps",
|
|
"io.sections", "io.sections.write", "io.sections.exec", "io.sections.readonly",
|
|
"dbg.stack", "dbg.heap", "dbg.map",
|
|
"dbg.maps", "dbg.maps.exec", "dbg.maps.write", "dbg.maps.readonly",
|
|
"anal.fcn", "anal.bb", NULL);
|
|
/* lines */
|
|
SETI ("lines.from", 0, "Start address for line seek");
|
|
SETCB ("lines.to", "$s", &cb_linesto, "End address for line seek");
|
|
SETCB ("lines.abs", "false", &cb_linesabs, "Enable absolute line numbers");
|
|
|
|
r_config_lock (cfg, true);
|
|
return true;
|
|
}
|