mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-07 13:51:16 +00:00
2614 lines
62 KiB
C
2614 lines
62 KiB
C
/* radare - LGPL - Copyright 2009-2016 - pancake */
|
|
|
|
#include "r_core.h"
|
|
|
|
#include <string.h>
|
|
|
|
#define MAX_FORMAT 3
|
|
|
|
enum {
|
|
R_BYTE_DATA = 1,
|
|
R_WORD_DATA = 2,
|
|
R_DWORD_DATA = 4,
|
|
R_QWORD_DATA = 8
|
|
};
|
|
|
|
typedef struct {
|
|
RCore *core;
|
|
int t_idx;
|
|
int t_ctr;
|
|
const char *type;
|
|
char *curname;
|
|
char *curfmt;
|
|
const char *optword;
|
|
} RCoreVisualTypes;
|
|
|
|
// TODO: move this helper into r_cons
|
|
static char *prompt(const char *str, const char *txt) {
|
|
char cmd[1024];
|
|
char *res = NULL;
|
|
char *oprompt = strdup (r_cons_singleton()->line->prompt);
|
|
r_cons_show_cursor (true);
|
|
if (txt && *txt) {
|
|
free (r_cons_singleton ()->line->contents);
|
|
r_cons_singleton ()->line->contents = strdup (txt);
|
|
} else {
|
|
free (r_cons_singleton ()->line->contents);
|
|
r_cons_singleton ()->line->contents = NULL;
|
|
}
|
|
*cmd = '\0';
|
|
r_line_set_prompt (str);
|
|
if (r_cons_fgets (cmd, sizeof (cmd)-1, 0, NULL) < 0) {
|
|
*cmd = '\0';
|
|
}
|
|
//line[strlen(line)-1]='\0';
|
|
if (*cmd) {
|
|
res = strdup (cmd);
|
|
}
|
|
r_line_set_prompt (oprompt);
|
|
free (oprompt);
|
|
free (r_cons_singleton ()->line->contents);
|
|
r_cons_singleton ()->line->contents = NULL;
|
|
return res;
|
|
}
|
|
|
|
static inline char *getformat (RCoreVisualTypes *vt, const char *k) {
|
|
return sdb_get (vt->core->anal->sdb_types,
|
|
sdb_fmt (0, "type.%s", k), 0);
|
|
}
|
|
|
|
static bool edit_bits (RCore *core) {
|
|
const int nbits = sizeof (ut64) * 8;
|
|
int i, j, x = 0;
|
|
RAsmOp asmop;
|
|
ut8 buf[sizeof (ut64)];
|
|
|
|
if (core->blocksize < sizeof (ut64)) {
|
|
return false;
|
|
}
|
|
memcpy (buf, core->block, sizeof (ut64));
|
|
for (;;) {
|
|
(void) r_asm_disassemble (core->assembler,
|
|
&asmop, buf, sizeof (ut64));
|
|
r_cons_clear00 ();
|
|
r_cons_printf ("r2's bit editor:\n\n");
|
|
r_cons_printf ("hex: %s\n", asmop.buf_hex);
|
|
r_cons_printf ("len: %d\n", asmop.size);
|
|
r_cons_printf ("asm: %s\n", asmop.buf_asm);
|
|
r_cons_printf ("chr:");
|
|
for (i = 0; i < 8; i++) {
|
|
ut8 *byte = buf + i;
|
|
char ch = IS_PRINTABLE(*byte)? *byte: '?';
|
|
r_cons_printf (" %5s'%c'", " ", ch);
|
|
}
|
|
r_cons_printf ("\ndec:");
|
|
for (i = 0; i < 8; i++) {
|
|
ut8 *byte = buf + i;
|
|
r_cons_printf (" %8d", *byte);
|
|
}
|
|
r_cons_printf ("\nhex:");
|
|
for (i = 0; i < 8; i++) {
|
|
ut8 *byte = buf + i;
|
|
r_cons_printf (" 0x%02x", *byte);
|
|
}
|
|
r_cons_printf ("\nbit: ");
|
|
for (i = 0; i < 8; i++) {
|
|
ut8 *byte = buf + i;
|
|
for (j = 0; j < 8; j++) {
|
|
bool bit = R_BIT_CHK (byte, 7 - j);
|
|
r_cons_printf ("%d", bit? 1: 0);
|
|
}
|
|
r_cons_print (" ");
|
|
}
|
|
r_cons_newline ();
|
|
char str_pos[128];
|
|
memset (str_pos, '-', nbits + 7);
|
|
str_pos[x + (x/8)] = '^';
|
|
str_pos[nbits + 7] = 0;
|
|
str_pos[8] = ' ';
|
|
str_pos[17] = ' ';
|
|
str_pos[26] = ' ';
|
|
str_pos[35] = ' ';
|
|
str_pos[44] = ' ';
|
|
str_pos[53] = ' ';
|
|
str_pos[62] = ' ';
|
|
r_cons_printf ("pos: %s\n", str_pos);
|
|
r_cons_newline ();
|
|
r_cons_visual_flush ();
|
|
|
|
int ch = r_cons_readchar ();
|
|
if (ch == -1 || ch == 4) {
|
|
break;
|
|
}
|
|
ch = r_cons_arrow_to_hjkl (ch); // get ESC+char, return 'hjkl' char
|
|
switch (ch) {
|
|
case 'q':
|
|
return false;
|
|
case 'j':
|
|
case 'k':
|
|
case ' ':
|
|
//togglebit();
|
|
{
|
|
const int nbyte = x / 8;
|
|
const int nbit = 7 - (x - (nbyte * 8));
|
|
ut8 *byte = buf + nbyte;
|
|
bool bit = R_BIT_CHK (byte, nbit);
|
|
if (bit) {
|
|
R_BIT_UNSET (byte, nbit);
|
|
} else {
|
|
R_BIT_SET (byte, nbit);
|
|
}
|
|
}
|
|
break;
|
|
case '+':
|
|
buf[(x/8)]++;
|
|
break;
|
|
case '-':
|
|
buf[(x/8)]--;
|
|
break;
|
|
case 'h':
|
|
x = R_MAX (x - 1, 0);
|
|
break;
|
|
case 'l':
|
|
x = R_MIN (x + 1, nbits - 1);
|
|
break;
|
|
case ':': // TODO: move this into a separate helper function
|
|
{
|
|
char cmd[1024];
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
cmd[0]='\0';
|
|
r_line_set_prompt (":> ");
|
|
if (r_cons_fgets (cmd, sizeof (cmd)-1, 0, NULL) < 0) {
|
|
cmd[0] = '\0';
|
|
}
|
|
r_core_cmd (core, cmd, 1);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
if (cmd[0]) {
|
|
r_cons_any_key (NULL);
|
|
}
|
|
r_cons_clear ();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// belongs to r_core_visual_types
|
|
static int sdbforcb (void *p, const char *k, const char *v) {
|
|
const char *pre = " ";
|
|
RCoreVisualTypes *vt = (RCoreVisualTypes*)p;
|
|
int use_color = vt->core->print->flags & R_PRINT_FLAGS_COLOR;
|
|
if (vt->optword) {
|
|
if (!strcmp (vt->type, "struct")) {
|
|
char *s = r_str_newf ("struct.%s.", vt->optword);
|
|
/* enum */
|
|
if (!strncmp (s, k, strlen (s))) {
|
|
if (vt->t_idx == vt->t_ctr) {
|
|
free (vt->curname);
|
|
vt->curname = strdup (k);
|
|
free (vt->curfmt);
|
|
vt->curfmt = strdup (v);
|
|
pre = ">";
|
|
}
|
|
if (use_color && *pre=='>')
|
|
r_cons_printf (Color_YELLOW" %s %s %s\n"
|
|
Color_RESET, pre, k+strlen (s), v);
|
|
else
|
|
r_cons_printf (" %s %s %s\n",
|
|
pre, k+strlen (s), v);
|
|
vt->t_ctr ++;
|
|
}
|
|
free (s);
|
|
} else {
|
|
char *s = r_str_newf ("%s.", vt->optword);
|
|
/* enum */
|
|
if (!strncmp (s, k, strlen (s))) {
|
|
if (!strstr (k, ".0x")) {
|
|
if (vt->t_idx == vt->t_ctr) {
|
|
free (vt->curname);
|
|
vt->curname = strdup (v);
|
|
free (vt->curfmt);
|
|
vt->curfmt = strdup (v);
|
|
pre = ">";
|
|
}
|
|
if (use_color && *pre=='>') {
|
|
r_cons_printf (Color_YELLOW" %s %s %s\n"
|
|
Color_RESET, pre, k, v);
|
|
} else {
|
|
r_cons_printf (" %s %s %s\n",
|
|
pre, k, v);
|
|
}
|
|
vt->t_ctr ++;
|
|
}
|
|
}
|
|
free (s);
|
|
}
|
|
} else if (!strcmp (v, vt->type)) {
|
|
if (!strcmp (vt->type, "type")) {
|
|
char *fmt = getformat (vt, k);
|
|
if (vt->t_idx == vt->t_ctr) {
|
|
free (vt->curname);
|
|
vt->curname = strdup (k);
|
|
free (vt->curfmt);
|
|
vt->curfmt = strdup (fmt);
|
|
pre = ">";
|
|
}
|
|
if (use_color && *pre=='>') {
|
|
r_cons_printf (Color_YELLOW" %s pf %3s %s\n"
|
|
Color_RESET,pre, fmt, k);
|
|
} else {
|
|
r_cons_printf (" %s pf %3s %s\n",
|
|
pre, fmt, k);
|
|
}
|
|
free (fmt);
|
|
} else {
|
|
if (vt->t_idx == vt->t_ctr) {
|
|
free (vt->curname);
|
|
vt->curname = strdup (k);
|
|
free (vt->curfmt);
|
|
vt->curfmt = strdup (v);
|
|
pre = ">";
|
|
}
|
|
if (use_color && *pre == '>') {
|
|
r_cons_printf (Color_YELLOW" %s %s\n"Color_RESET,
|
|
(vt->t_idx == vt->t_ctr)?
|
|
">":" ", k);
|
|
} else {
|
|
r_cons_printf (" %s %s\n",
|
|
(vt->t_idx == vt->t_ctr)?
|
|
">":" ", k);
|
|
}
|
|
}
|
|
vt->t_ctr ++;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
R_API int r_core_visual_types(RCore *core) {
|
|
RCoreVisualTypes vt = {core, 0, 0};
|
|
int i, j, ch;
|
|
int _option = 0;
|
|
int option = 0;
|
|
char *txt;
|
|
char cmd[1024];
|
|
int menu = 0;
|
|
int h_opt = 0;
|
|
char *optword = NULL;
|
|
const char *opts[] = {
|
|
"type",
|
|
"enum",
|
|
"struct",
|
|
"func",
|
|
"union",
|
|
NULL
|
|
};
|
|
bool use_color = core->print->flags & R_PRINT_FLAGS_COLOR;
|
|
for (j = i = 0; i < R_FLAG_SPACES_MAX; i++) {
|
|
if (core->flags->spaces[i]) {
|
|
j = 1;
|
|
}
|
|
}
|
|
if (j == 0) {
|
|
menu = 1;
|
|
}
|
|
for (;;) {
|
|
r_cons_clear00 ();
|
|
for (i = 0; opts[i]; i++) {
|
|
const char *fmt = use_color
|
|
? (h_opt == i)
|
|
? Color_BGREEN"[%s] "Color_RESET
|
|
: Color_GREEN" %s "Color_RESET
|
|
: (h_opt == i)
|
|
? "[%s] "
|
|
: " %s ";
|
|
r_cons_printf (fmt, opts[i]);
|
|
}
|
|
r_cons_newline ();
|
|
if (optword) {
|
|
r_cons_printf (">> %s\n", optword);
|
|
}
|
|
vt.t_idx = option;
|
|
vt.t_ctr = 0;
|
|
vt.type = opts[h_opt];
|
|
vt.optword = optword;
|
|
sdb_foreach (core->anal->sdb_types, sdbforcb, &vt);
|
|
|
|
r_cons_visual_flush ();
|
|
ch = r_cons_readchar ();
|
|
if (ch == -1 || ch == 4) {
|
|
return false;
|
|
}
|
|
ch = r_cons_arrow_to_hjkl (ch); // get ESC+char, return 'hjkl' char
|
|
switch (ch) {
|
|
case 'h':
|
|
h_opt--;
|
|
if (h_opt<0)
|
|
h_opt = 0;
|
|
option = 0;
|
|
R_FREE (optword);
|
|
break;
|
|
case 'l':
|
|
h_opt++;
|
|
option = 0;
|
|
if (!opts[h_opt])
|
|
h_opt--;
|
|
R_FREE (optword);
|
|
break;
|
|
case 'o':
|
|
{
|
|
char *file = prompt ("Filename: ", NULL);
|
|
if (file) {
|
|
r_core_cmdf (core, "\"to %s\"", file);
|
|
free (file);
|
|
}
|
|
}
|
|
break;
|
|
case 'j':
|
|
if (++option >= vt.t_ctr) {
|
|
option = vt.t_ctr - 1;
|
|
}
|
|
break;
|
|
case 'J':
|
|
option += 10;
|
|
if (option >= vt.t_ctr) {
|
|
option = vt.t_ctr-1;
|
|
}
|
|
break;
|
|
case 'k':
|
|
if (--option < 0) {
|
|
option = 0;
|
|
}
|
|
break;
|
|
case 'K':
|
|
option -= 10;
|
|
if (option < 0) {
|
|
option = 0;
|
|
}
|
|
break;
|
|
case 'b':
|
|
r_core_cmdf (core, "tl %s", vt.curname);
|
|
break;
|
|
case -1: // EOF
|
|
case 'q':
|
|
if (optword) {
|
|
R_FREE (optword);
|
|
break;
|
|
}
|
|
if (menu <= 0) {
|
|
return true;
|
|
}
|
|
menu--;
|
|
option = _option;
|
|
if (menu==0) {
|
|
// if no flagspaces, just quit
|
|
for (j = i = 0; i < R_FLAG_SPACES_MAX; i++) {
|
|
if (core->flags->spaces[i]) {
|
|
j = 1;
|
|
}
|
|
}
|
|
if (!j) {
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
case 'a':
|
|
{
|
|
txt = prompt ("add C type: ", NULL);
|
|
if (txt) {
|
|
r_core_cmdf (core, "\"td %s\"", txt);
|
|
free (txt);
|
|
}
|
|
}
|
|
break;
|
|
case 'd':
|
|
r_core_cmdf (core, "t- %s", vt.curname);
|
|
break;
|
|
case '-':
|
|
r_core_cmd0 (core, "to -");
|
|
break;
|
|
case ' ':
|
|
case '\r':
|
|
case '\n':
|
|
case 'e':
|
|
if (optword) {
|
|
/* TODO: edit field */
|
|
} else {
|
|
switch (h_opt) {
|
|
case 0: // type
|
|
/* TODO: do something with this data */
|
|
prompt ("name: ", vt.curname);
|
|
prompt ("pf: ", vt.curfmt);
|
|
break;
|
|
case 1: // enum
|
|
case 2: // struct
|
|
free (optword);
|
|
optword = strdup (vt.curname);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case '?':
|
|
r_cons_clear00 ();
|
|
r_cons_printf (
|
|
"Vt?: Visual Types Help:\n\n"
|
|
" q - quit menu\n"
|
|
" j/k - down/up keys\n"
|
|
" h/l - left-right\n"
|
|
" a - add new type (C syntax)\n"
|
|
" b - bind type to current offset\n"
|
|
" d - delete current type\n"
|
|
" e - edit current type\n"
|
|
" o - open .h include file\n"
|
|
" - - Open cfg.editor to load types\n"
|
|
" : - enter command\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case ':':
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
cmd[0]='\0';
|
|
r_line_set_prompt (":> ");
|
|
if (r_cons_fgets (cmd, sizeof (cmd)-1, 0, NULL) < 0) {
|
|
cmd[0]='\0';
|
|
}
|
|
r_core_cmd (core, cmd, 1);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
if (cmd[0]) {
|
|
r_cons_any_key (NULL);
|
|
}
|
|
r_cons_clear ();
|
|
continue;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int cmtcb(void *usr, const char *k, const char *v) {
|
|
if (!strncmp (k, "meta.C.", 7)) {
|
|
RList *list = (RList*)usr;
|
|
char *msg, *comma = strchr (v, ',');
|
|
if (comma) {
|
|
comma = strchr (comma + 1, ',');
|
|
if (comma) {
|
|
msg = (char *)sdb_decode (comma + 1, NULL);
|
|
if (msg) {
|
|
msg = r_str_replace (msg, "\n", "", true);
|
|
r_list_append (list, r_str_newf ("%s %s", k+7, msg));
|
|
free (msg);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
R_API bool r_core_visual_hudstuff(RCore *core) {
|
|
RListIter *iter;
|
|
RFlagItem *flag;
|
|
ut64 addr;
|
|
char *res;
|
|
RList *list = r_list_new ();
|
|
if (!list) {
|
|
return false;
|
|
}
|
|
list->free = free;
|
|
r_list_foreach (core->flags->flags, iter, flag) {
|
|
r_list_append (list, r_str_newf ("0x%08"PFMT64x" %s",
|
|
flag->offset, flag->name));
|
|
}
|
|
sdb_foreach (core->anal->sdb_meta, cmtcb, list);
|
|
res = r_cons_hud (list, NULL, r_config_get_i (core->config, "scr.color"));
|
|
if (res) {
|
|
char *p = strchr (res, ' ');
|
|
if (p) {
|
|
*p = 0;
|
|
}
|
|
addr = r_num_get (NULL, res);
|
|
r_core_seek (core, addr, true);
|
|
free (res);
|
|
}
|
|
r_list_free (list);
|
|
return res? true: false;
|
|
}
|
|
|
|
static bool r_core_visual_config_hud(RCore *core) {
|
|
RListIter *iter;
|
|
RConfigNode *bt;
|
|
RList *list = r_list_new ();
|
|
if (!list) return false;
|
|
char *res;
|
|
list->free = free;
|
|
r_list_foreach (core->config->nodes, iter, bt) {
|
|
r_list_append (list, r_str_newf ("%s %s", bt->name, bt->value));
|
|
}
|
|
res = r_cons_hud (list, NULL, r_config_get_i (core->config, "scr.color"));
|
|
if (res) {
|
|
const char *oldvalue = NULL;
|
|
char cmd[512];
|
|
char *p = strchr (res, ' ');
|
|
if (p) *p = 0;
|
|
oldvalue = r_config_get (core->config, res);
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
cmd[0] = '\0';
|
|
eprintf ("set new value for %s (old=%s)\n", res, oldvalue);
|
|
r_line_set_prompt (":> ");
|
|
if (r_cons_fgets (cmd, sizeof (cmd) - 1, 0, NULL) < 0)
|
|
cmd[0]='\0';
|
|
r_config_set (core->config, res, cmd);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
r_list_free (list);
|
|
return true;
|
|
}
|
|
|
|
// TODO: skip N first elements
|
|
// TODO: show only N elements of the list
|
|
// TODO: wrap index when out of boundaries
|
|
// TODO: Add support to show class fields too
|
|
static void *show_class(RCore *core, int mode, int idx, RBinClass *_c, const char *grep) {
|
|
bool show_color = r_config_get_i (core->config, "scr.color");
|
|
RListIter *iter;
|
|
RBinClass *c, *cur = NULL;
|
|
RBinSymbol *m, *mur = NULL;
|
|
RList *list;
|
|
int i = 0;
|
|
int skip = idx - 10;
|
|
|
|
switch (mode) {
|
|
case 'c':
|
|
r_cons_printf ("Classes:\n\n");
|
|
list = r_bin_get_classes (core->bin);
|
|
r_list_foreach (list, iter, c) {
|
|
if (grep) {
|
|
if (!r_str_casestr (c->name, grep)) {
|
|
i++;
|
|
continue;
|
|
}
|
|
} else {
|
|
if (idx > 10) {
|
|
skip--;
|
|
if (skip > 0) {
|
|
i++;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
if (show_color) {
|
|
if (i == idx) {
|
|
const char *clr = Color_BLUE;
|
|
r_cons_printf (Color_GREEN ">>" Color_RESET " %02d %s0x%08"
|
|
PFMT64x Color_YELLOW " %s\n" Color_RESET,
|
|
i, clr, c->addr, c->name);
|
|
} else {
|
|
r_cons_printf ("- %02d %s0x%08"PFMT64x Color_RESET" %s\n",
|
|
i, core->cons->pal.offset, c->addr, c->name);
|
|
}
|
|
} else {
|
|
r_cons_printf ("%s %02d 0x%08"PFMT64x" %s\n",
|
|
(i==idx)?">>":"- ", i, c->addr, c->name);
|
|
}
|
|
if (i++ == idx) {
|
|
cur = c;
|
|
}
|
|
}
|
|
return cur;
|
|
case 'f':
|
|
// show fields
|
|
r_cons_printf ("Fields:\n\n");
|
|
break;
|
|
case 'm':
|
|
// show methods
|
|
if (!_c) {
|
|
eprintf ("No class defined\n");
|
|
return mur;
|
|
}
|
|
r_cons_printf ("MethodsFor: %s\n\n", _c->name);
|
|
r_list_foreach (_c->methods, iter, m) {
|
|
if (grep) {
|
|
if (!r_str_casestr (m->name, grep)) {
|
|
i++;
|
|
continue;
|
|
}
|
|
} else {
|
|
if (idx > 10) {
|
|
skip--;
|
|
if (skip > 0) {
|
|
i++;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
if (show_color) {
|
|
if (i == idx) {
|
|
const char *clr = Color_BLUE;
|
|
r_cons_printf (Color_GREEN ">>" Color_RESET " %02d %s0x%08"
|
|
PFMT64x Color_YELLOW " %s\n" Color_RESET,
|
|
i, clr, m->vaddr, m->name);
|
|
} else {
|
|
r_cons_printf ("- %02d %s0x%08"PFMT64x Color_RESET" %s\n",
|
|
i, core->cons->pal.offset, m->vaddr, m->name);
|
|
}
|
|
} else {
|
|
r_cons_printf ("%s %02d 0x%08"PFMT64x" %s\n",
|
|
(i==idx)? ">>": "- ", i, m->vaddr, m->name);
|
|
}
|
|
if (i++ == idx) {
|
|
mur = m;
|
|
}
|
|
}
|
|
return mur;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
R_API int r_core_visual_classes(RCore *core) {
|
|
int ch, option = 0;
|
|
char cmd[1024];
|
|
int mode = 'c';
|
|
RBinClass *cur = NULL;
|
|
RBinSymbol *mur = NULL;
|
|
void *ptr;
|
|
int oldcur = 0;
|
|
char *grep = NULL;
|
|
bool grepmode = false;
|
|
|
|
for (;;) {
|
|
int cols;
|
|
r_cons_clear00 ();
|
|
if (grepmode) {
|
|
r_cons_printf ("Grep: %s\n", grep? grep: "");
|
|
}
|
|
ptr = show_class (core, mode, option, cur, grep);
|
|
switch (mode) {
|
|
case 'm':
|
|
mur = (RBinSymbol*)ptr;
|
|
break;
|
|
case 'c':
|
|
cur = (RBinClass*)ptr;
|
|
break;
|
|
}
|
|
|
|
/* update terminal size */
|
|
(void) r_cons_get_size (&cols);
|
|
r_cons_visual_flush ();
|
|
ch = r_cons_readchar ();
|
|
if (ch==-1 || ch==4) {
|
|
return false;
|
|
}
|
|
|
|
if (grepmode) {
|
|
switch (ch) {
|
|
case 127:
|
|
if (grep) {
|
|
int len = strlen (grep);
|
|
if (len < 1) {
|
|
grepmode = false;
|
|
} else {
|
|
grep[len - 1] = 0;
|
|
}
|
|
}
|
|
break;
|
|
case ' ':
|
|
case '\r':
|
|
case '\n':
|
|
R_FREE (grep);
|
|
grepmode = false;
|
|
break;
|
|
default:
|
|
grep = grep
|
|
? r_str_concatf (grep, "%c", ch)
|
|
: r_str_newf ("%c", ch);
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
ch = r_cons_arrow_to_hjkl (ch); // get ESC+char, return 'hjkl' char
|
|
switch (ch) {
|
|
case 'C':
|
|
r_config_toggle (core->config, "scr.color");
|
|
break;
|
|
case 'J': option += 10; break;
|
|
case 'j': option++; break;
|
|
case 'k': if (--option < 0) option = 0; break;
|
|
case 'K': option -= 10; if (option<0) option = 0; break;
|
|
case 'g':
|
|
{
|
|
char *num = prompt ("Index:", NULL);
|
|
option = atoi (num);
|
|
free (num);
|
|
}
|
|
break;
|
|
case 'p':
|
|
if (mode == 'm' && mur) {
|
|
r_core_seek (core, mur->vaddr, true);
|
|
r_core_cmd0 (core, "af;pdf~..");
|
|
}
|
|
break;
|
|
case 'h':
|
|
case 127: // backspace
|
|
case 'b': // back
|
|
case 'q':
|
|
if (mode == 'c') {
|
|
return true;
|
|
}
|
|
mode = 'c';
|
|
option = oldcur;
|
|
break;
|
|
case '/':
|
|
grepmode = true;
|
|
break;
|
|
case 'l':
|
|
case ' ':
|
|
case '\r':
|
|
case '\n':
|
|
if (mur && mode == 'm') {
|
|
r_core_seek (core, mur->vaddr, true);
|
|
return true;
|
|
}
|
|
if (cur) {
|
|
oldcur = option;
|
|
option = 0;
|
|
mode = 'm';
|
|
}
|
|
break;
|
|
case '?':
|
|
r_cons_clear00 ();
|
|
r_cons_printf (
|
|
"\nVF: Visual Classes help:\n\n"
|
|
" q - quit menu\n"
|
|
" j/k - down/up keys\n"
|
|
" h/b - go back\n"
|
|
" / - grep mode\n"
|
|
" C - toggle colors\n"
|
|
" l/' ' - accept current selection\n"
|
|
" p - preview method disasm with less\n"
|
|
" : - enter command\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case ':':
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
cmd[0] = '\0';
|
|
r_line_set_prompt (":> ");
|
|
if (r_cons_fgets (cmd, sizeof (cmd)-1, 0, NULL) < 0) {
|
|
cmd[0]='\0';
|
|
}
|
|
//line[strlen(line)-1]='\0';
|
|
r_core_cmd (core, cmd, 1);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
if (cmd[0]) {
|
|
r_cons_any_key (NULL);
|
|
}
|
|
//cons_gotoxy(0,0);
|
|
r_cons_clear ();
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API int r_core_visual_trackflags(RCore *core) {
|
|
const char *fs = NULL, *fs2 = NULL;
|
|
int hit, i, j, ch;
|
|
RListIter *iter;
|
|
RFlagItem *flag;
|
|
int _option = 0;
|
|
int option = 0;
|
|
char cmd[1024];
|
|
int format = 0;
|
|
int delta = 7;
|
|
int menu = 0;
|
|
|
|
for (j=i=0; i<R_FLAG_SPACES_MAX; i++)
|
|
if (core->flags->spaces[i])
|
|
j = 1;
|
|
if (j==0) menu = 1;
|
|
for (;;) {
|
|
r_cons_clear00 ();
|
|
|
|
if (menu) {
|
|
r_cons_printf ("Flags in flagspace '%s'. Press '?' for help.\n\n",
|
|
(core->flags->space_idx==-1)?"*":core->flags->spaces[core->flags->space_idx]);
|
|
hit = 0;
|
|
i = j = 0;
|
|
r_list_foreach (core->flags->flags, iter, flag) {
|
|
/* filter per flag spaces */
|
|
if ((core->flags->space_idx != -1) &&
|
|
(flag->space != core->flags->space_idx)) {
|
|
continue;
|
|
}
|
|
if (option == i) {
|
|
fs2 = flag->name;
|
|
hit = 1;
|
|
}
|
|
if ((i>=option-delta) && ((i<option+delta)||((option<delta)&&(i<(delta<<1))))) {
|
|
r_cons_printf (" %c %03d 0x%08"PFMT64x" %4"PFMT64d" %s\n",
|
|
(option==i)?'>':' ',
|
|
i, flag->offset, flag->size, flag->name);
|
|
j++;
|
|
}
|
|
i++;
|
|
}
|
|
if (!hit && i > 0) {
|
|
option = i - 1;
|
|
continue;
|
|
}
|
|
if (fs2) {
|
|
int cols, rows = r_cons_get_size (&cols);
|
|
//int rows = 20;
|
|
rows -= 12;
|
|
r_cons_printf ("\n Selected: %s\n\n", fs2);
|
|
// Honor MAX_FORMATS here
|
|
switch (format) {
|
|
case 0: snprintf (cmd, sizeof (cmd), "px %d @ %s!64", rows*16, fs2); core->printidx = 0; break;
|
|
case 1: snprintf (cmd, sizeof (cmd), "pd %d @ %s!64", rows, fs2); core->printidx = 1; break;
|
|
case 2: snprintf (cmd, sizeof (cmd), "ps @ %s!64", fs2); core->printidx = 5; break;
|
|
case 3: strcpy (cmd, "f="); break;
|
|
default: format = 0; continue;
|
|
}
|
|
if (*cmd) r_core_cmd (core, cmd, 0);
|
|
} else r_cons_printf ("(no flags)\n");
|
|
} else {
|
|
r_cons_printf ("Flag spaces:\n\n");
|
|
hit = 0;
|
|
for (j=i=0;i<R_FLAG_SPACES_MAX;i++) {
|
|
if (core->flags->spaces[i]) {
|
|
if (option == i) {
|
|
fs = core->flags->spaces[i];
|
|
hit = 1;
|
|
}
|
|
if ((i >= option - delta) && ((i < option + delta)|| \
|
|
((option < delta) && (i < (delta << 1))))) {
|
|
r_cons_printf(" %c %02d %c %s\n",
|
|
(option==i)?'>':' ', j,
|
|
(i==core->flags->space_idx)?'*':' ',
|
|
core->flags->spaces[i]);
|
|
j++;
|
|
}
|
|
}
|
|
}
|
|
if (core->flags->spaces[9]) {
|
|
if (option == j) {
|
|
fs = "*";
|
|
hit = 1;
|
|
}
|
|
r_cons_printf (" %c %02d %c %s\n",
|
|
(option==j)?'>':' ', j,
|
|
(i==core->flags->space_idx)?'*':' ',
|
|
"*");
|
|
}
|
|
if (!hit && j > 0) {
|
|
option = j - 1;
|
|
continue;
|
|
}
|
|
}
|
|
r_cons_visual_flush ();
|
|
ch = r_cons_readchar ();
|
|
if (ch == -1 || ch == 4) {
|
|
return false;
|
|
}
|
|
ch = r_cons_arrow_to_hjkl (ch); // get ESC+char, return 'hjkl' char
|
|
switch (ch) {
|
|
case 'C':
|
|
r_config_toggle (core->config, "scr.color");
|
|
break;
|
|
case '_':
|
|
if (r_core_visual_hudstuff (core))
|
|
return true;
|
|
break;
|
|
case 'J': option += 10; break;
|
|
case 'o': r_flag_sort (core->flags, 0); break;
|
|
case 'n': r_flag_sort (core->flags, 1); break;
|
|
case 'j': option++; break;
|
|
case 'k': if (--option<0) option = 0; break;
|
|
case 'K': option-=10; if (option<0) option = 0; break;
|
|
case 'h':
|
|
case 'b': // back
|
|
case 'q':
|
|
if (menu <= 0) return true;
|
|
menu--;
|
|
option = _option;
|
|
if (menu == 0) {
|
|
// if no flagspaces, just quit
|
|
for (j=i=0;i<R_FLAG_SPACES_MAX;i++) {
|
|
if (core->flags->spaces[i]) {
|
|
j = 1;
|
|
}
|
|
}
|
|
if (!j) {
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
case 'a':
|
|
switch (menu) {
|
|
case 0: // new flag space
|
|
r_cons_show_cursor (true);
|
|
r_line_set_prompt ("add flagspace: ");
|
|
strcpy (cmd, "fs ");
|
|
if (r_cons_fgets (cmd+3, sizeof (cmd)-4, 0, NULL) > 0) {
|
|
r_core_cmd (core, cmd, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
break;
|
|
case 1: // new flag
|
|
r_cons_show_cursor (true);
|
|
r_line_set_prompt ("add flag: ");
|
|
strcpy (cmd, "f ");
|
|
if (r_cons_fgets (cmd+2, sizeof (cmd)-3, 0, NULL) > 0) {
|
|
r_core_cmd (core, cmd, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
case 'd':
|
|
r_flag_unset_name (core->flags, fs2);
|
|
break;
|
|
case 'e':
|
|
/* TODO: prompt for addr, size, name */
|
|
eprintf ("TODO\n");
|
|
r_sys_sleep (1);
|
|
break;
|
|
case '*':
|
|
r_core_block_size (core, core->blocksize+16);
|
|
break;
|
|
case '/':
|
|
r_core_block_size (core, core->blocksize-16);
|
|
break;
|
|
case '+':
|
|
if (menu==1)
|
|
r_core_cmdf (core, "f %s=%s+1", fs2, fs2);
|
|
else r_core_block_size (core, core->blocksize+1);
|
|
break;
|
|
case '-':
|
|
if (menu == 1) {
|
|
r_core_cmdf (core, "f %s=%s-1", fs2, fs2);
|
|
} else {
|
|
r_core_block_size (core, core->blocksize-1);
|
|
}
|
|
break;
|
|
case 'r': // "Vtr"
|
|
if (menu == 1) {
|
|
int len;
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
// TODO: use r_flag_rename or wtf?..fr doesnt uses this..
|
|
snprintf (cmd, sizeof (cmd), "fr %s ", fs2);
|
|
len = strlen (cmd);
|
|
eprintf ("Rename flag '%s' as:\n", fs2);
|
|
r_line_set_prompt (":> ");
|
|
if (r_cons_fgets (cmd + len, sizeof (cmd) - len - 1, 0, NULL) < 0) {
|
|
cmd[0] = '\0';
|
|
}
|
|
r_core_cmd (core, cmd, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
break;
|
|
case 'R':
|
|
if (menu == 1) {
|
|
char line[1024];
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
eprintf ("Rename function '%s' as:\n", fs2);
|
|
r_line_set_prompt (":> ");
|
|
if (r_cons_fgets (line, sizeof (line), 0, NULL) < 0) {
|
|
cmd[0]='\0';
|
|
}
|
|
snprintf (cmd, sizeof (cmd), "afr %s %s", line, fs2);
|
|
r_core_cmd (core, cmd, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
break;
|
|
case 'P': if (--format<0) format = MAX_FORMAT; break;
|
|
// = (format<=0)? MAX_FORMAT: format-1; break;
|
|
case 'p': format++; break;
|
|
case 'l':
|
|
case ' ':
|
|
case '\r':
|
|
case '\n':
|
|
if (menu == 1) {
|
|
sprintf (cmd, "s %s", fs2);
|
|
r_core_cmd (core, cmd, 0);
|
|
return true;
|
|
}
|
|
r_flag_space_set (core->flags, fs);
|
|
menu = 1;
|
|
_option = option;
|
|
option = 0;
|
|
break;
|
|
case '?':
|
|
r_cons_clear00 ();
|
|
r_cons_printf (
|
|
"\nVF: Visual Flags help:\n\n"
|
|
" q - quit menu\n"
|
|
" j/k - line down/up keys\n"
|
|
" J/K - page down/up keys\n"
|
|
" h/b - go back\n"
|
|
" C - toggle colors\n"
|
|
" l/' ' - accept current selection\n"
|
|
" a/d/e - add/delete/edit flag\n"
|
|
" +/- - increase/decrease block size\n"
|
|
" o - sort flags by offset\n"
|
|
" r/R - rename flag / Rename function\n"
|
|
" n - sort flags by name\n"
|
|
" p/P - rotate print format\n"
|
|
" _ - hud for flags and comments\n"
|
|
" : - enter command\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case ':':
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
cmd[0]='\0';
|
|
r_line_set_prompt (":> ");
|
|
if (r_cons_fgets (cmd, sizeof (cmd)-1, 0, NULL) <0)
|
|
cmd[0]='\0';
|
|
//line[strlen(line)-1]='\0';
|
|
r_core_cmd (core, cmd, 1);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
if (cmd[0])
|
|
r_cons_any_key (NULL);
|
|
//cons_gotoxy(0,0);
|
|
r_cons_clear ();
|
|
continue;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
R_API int r_core_visual_comments (RCore *core) {
|
|
#undef DB
|
|
#define DB core->anal->sdb_meta
|
|
const char *val, *comma = NULL;
|
|
char *list = sdb_get (DB, "meta.C", 0);
|
|
char *str, *next, *cur = list;
|
|
char key[128], cmd[512], *p = NULL;
|
|
int i, ch, option = 0, delta = 7;
|
|
int format = 0, found = 0;
|
|
ut64 addr, from = 0, size = 0;
|
|
|
|
for (;;) {
|
|
r_cons_clear00 ();
|
|
r_cons_strcat ("Comments:\n");
|
|
i = 0;
|
|
found = 0;
|
|
if (list) {
|
|
for (i=0; ;i++) {
|
|
cur = sdb_anext (cur, &next);
|
|
addr = sdb_atoi (cur);
|
|
snprintf (key, sizeof (key)-1, "meta.C.0x%08"PFMT64x, addr);
|
|
val = sdb_const_get (DB, key, 0);
|
|
if (val)
|
|
comma = strchr (val, ',');
|
|
if (comma) {
|
|
str = (char *)sdb_decode (comma+1, 0);
|
|
if ((i>=option-delta) && ((i<option+delta)||((option<delta)&&(i<(delta<<1))))) {
|
|
r_str_sanitize (str);
|
|
if (option==i) {
|
|
found = 1;
|
|
from = addr;
|
|
size = 1; // XXX: remove this thing size for comments is useless d->size;
|
|
free (p);
|
|
p = str;
|
|
r_cons_printf (" > %s\n", str);
|
|
} else {
|
|
r_cons_printf (" %s\n", str);
|
|
free (str);
|
|
}
|
|
} else free (str);
|
|
}
|
|
if (!next)
|
|
break;
|
|
cur = next;
|
|
}
|
|
}
|
|
|
|
if (!found) {
|
|
if (--option < 0) {
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
r_cons_newline ();
|
|
|
|
switch (format) {
|
|
case 0: sprintf (cmd, "px @ 0x%"PFMT64x":64", from); core->printidx = 0; break;
|
|
case 1: sprintf (cmd, "pd 12 @ 0x%"PFMT64x":64", from); core->printidx = 1; break;
|
|
case 2: sprintf (cmd, "ps @ 0x%"PFMT64x":64", from); core->printidx = 5; break;
|
|
default: format = 0; continue;
|
|
}
|
|
if (*cmd) {
|
|
r_core_cmd (core, cmd, 0);
|
|
}
|
|
r_cons_visual_flush ();
|
|
ch = r_cons_readchar ();
|
|
ch = r_cons_arrow_to_hjkl (ch); // get ESC+char, return 'hjkl' char
|
|
switch (ch) {
|
|
case 'a':
|
|
//TODO
|
|
break;
|
|
case 'e':
|
|
//TODO
|
|
break;
|
|
case 'd':
|
|
if (p)
|
|
r_meta_del (core->anal, R_META_TYPE_ANY, from, size, p);
|
|
break;
|
|
case 'P':
|
|
if (--format<0)
|
|
format = MAX_FORMAT;
|
|
break;
|
|
case 'p':
|
|
format++;
|
|
break;
|
|
case 'J':
|
|
option += 10;
|
|
break;
|
|
case 'j':
|
|
option++;
|
|
break;
|
|
case 'k':
|
|
if (--option < 0) {
|
|
option = 0;
|
|
}
|
|
break;
|
|
case 'K':
|
|
option -= 10;
|
|
if (option < 0) {
|
|
option = 0;
|
|
}
|
|
break;
|
|
case 'l':
|
|
case ' ':
|
|
case '\r':
|
|
case '\n':
|
|
r_core_cmdf (core, "s 0x%"PFMT64x, from);
|
|
R_FREE (p);
|
|
return true;
|
|
case 'q':
|
|
R_FREE (p);
|
|
return true;
|
|
case '?':
|
|
case 'h':
|
|
r_cons_clear00 ();
|
|
r_cons_printf (
|
|
"\nVT: Visual Comments/Anal help:\n\n"
|
|
" q - quit menu\n"
|
|
" j/k - down/up keys\n"
|
|
" h/b - go back\n"
|
|
" l/' ' - accept current selection\n"
|
|
" a/d/e - add/delete/edit comment/anal symbol\n"
|
|
" p/P - rotate print format\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
}
|
|
if (p) {
|
|
R_FREE (p);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void config_visual_hit_i(RCore *core, const char *name, int delta) {
|
|
struct r_config_node_t *node;
|
|
node = r_config_node_get (core->config, name);
|
|
if (node && ((node->flags & CN_INT) || (node->flags & CN_OFFT))) {
|
|
r_config_set_i(core->config, name, r_config_get_i(core->config, name)+delta);
|
|
}
|
|
}
|
|
|
|
/* Visually activate the config variable */
|
|
static void config_visual_hit(RCore *core, const char *name, int editor) {
|
|
char buf[1024];
|
|
RConfigNode *node;
|
|
|
|
if (!(node = r_config_node_get (core->config, name)))
|
|
return;
|
|
if (node->flags & CN_BOOL) {
|
|
r_config_set_i (core->config, name, node->i_value? 0:1);
|
|
} else {
|
|
// XXX: must use config_set () to run callbacks!
|
|
if (editor) {
|
|
char * buf = r_core_editor (core, NULL, node->value);
|
|
node->value = r_str_dup (node->value, buf);
|
|
free (buf);
|
|
} else {
|
|
// FGETS AND SO
|
|
r_cons_printf ("New value (old=%s): \n", node->value);
|
|
r_cons_show_cursor (true);
|
|
r_cons_flush ();
|
|
r_cons_set_raw (0);
|
|
r_line_set_prompt (":> ");
|
|
r_cons_fgets (buf, sizeof (buf)-1, 0, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
r_config_set (core->config, name, buf);
|
|
//node->value = r_str_dup (node->value, buf);
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API void r_core_visual_config(RCore *core) {
|
|
char *fs = NULL, *fs2 = NULL, *desc = NULL;
|
|
int i, j, ch, hit, show;
|
|
int option, _option = 0;
|
|
RListIter *iter;
|
|
RConfigNode *bt;
|
|
char old[1024];
|
|
int delta = 9;
|
|
int menu = 0;
|
|
old[0]='\0';
|
|
|
|
option = 0;
|
|
for (;;) {
|
|
r_cons_clear00 ();
|
|
r_cons_get_size (&delta);
|
|
delta /= 4;
|
|
|
|
switch (menu) {
|
|
case 0: // flag space
|
|
r_cons_printf ("[EvalSpace]\n\n");
|
|
hit = j = i = 0;
|
|
r_list_foreach (core->config->nodes, iter, bt) {
|
|
if (option==i) {
|
|
fs = bt->name;
|
|
}
|
|
if (old[0]=='\0') {
|
|
r_str_ccpy (old, bt->name, '.');
|
|
show = 1;
|
|
} else if (r_str_ccmp (old, bt->name, '.')) {
|
|
r_str_ccpy (old, bt->name, '.');
|
|
show = 1;
|
|
} else show = 0;
|
|
|
|
if (show) {
|
|
if (option == i) hit = 1;
|
|
if ( (i >=option-delta) && ((i<option+delta)||((option<delta)&&(i<(delta<<1))))) {
|
|
r_cons_printf(" %c %s\n", (option==i)?'>':' ', old);
|
|
j++;
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
if (!hit && j>0) {
|
|
option--;
|
|
continue;
|
|
}
|
|
r_cons_printf ("\n Sel:%s \n\n", fs);
|
|
break;
|
|
case 1: // flag selection
|
|
r_cons_printf ("[EvalSpace < Variables: %s]\n\n", fs);
|
|
hit = 0;
|
|
j = i = 0;
|
|
// TODO: cut -d '.' -f 1 | sort | uniq !!!
|
|
r_list_foreach (core->config->nodes, iter, bt) {
|
|
if (!r_str_ccmp (bt->name, fs, '.')) {
|
|
if (option==i) {
|
|
fs2 = bt->name;
|
|
desc = bt->desc;
|
|
hit = 1;
|
|
}
|
|
if ( (i>=option-delta) && ((i<option+delta)||((option<delta)&&(i<(delta<<1))))) {
|
|
// TODO: Better align
|
|
r_cons_printf (" %c %s = %s\n", (option==i)?'>':' ', bt->name, bt->value);
|
|
j++;
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
if (!hit && j>0) {
|
|
option = i-1;
|
|
continue;
|
|
}
|
|
if (fs2 != NULL)
|
|
// TODO: Break long lines.
|
|
r_cons_printf ("\n Selected: %s (%s)\n\n",
|
|
fs2, desc);
|
|
}
|
|
|
|
if (fs && !strncmp (fs, "asm.", 4)) {
|
|
r_core_cmd (core, "pd $r", 0);
|
|
}
|
|
r_cons_visual_flush ();
|
|
ch = r_cons_readchar ();
|
|
if (ch==4 || ch==-1)
|
|
return;
|
|
ch = r_cons_arrow_to_hjkl (ch); // get ESC+char, return 'hjkl' char
|
|
|
|
switch (ch) {
|
|
case 'j': option++; break;
|
|
case 'k': option = (option<=0)? 0: option-1; break;
|
|
case 'J': option+=4; break;
|
|
case 'K': option = (option<=3)? 0: option-4; break;
|
|
case 'h':
|
|
case 'b': // back
|
|
menu = 0;
|
|
option = _option;
|
|
break;
|
|
case '_':
|
|
r_core_visual_config_hud (core);
|
|
break;
|
|
case 'q':
|
|
if (menu <= 0) {
|
|
return;
|
|
}
|
|
menu--;
|
|
option = _option;
|
|
break;
|
|
case '$':
|
|
r_core_cmd0 (core, "?$");
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case '*':
|
|
case '+':
|
|
fs2 ? config_visual_hit_i (core, fs2, +1) : 0;
|
|
continue;
|
|
case '/':
|
|
case '-':
|
|
fs2 ? config_visual_hit_i (core, fs2, -1) : 0;
|
|
continue;
|
|
case 'l':
|
|
case 'E': // edit value
|
|
case 'e': // edit value
|
|
case ' ':
|
|
case '\r':
|
|
case '\n': // never happens
|
|
if (menu == 1) {
|
|
fs2 ? config_visual_hit (core, fs2, (ch=='E')) : 0;
|
|
} else {
|
|
menu = 1;
|
|
_option = option;
|
|
option = 0;
|
|
}
|
|
break;
|
|
case '?':
|
|
r_cons_clear00 ();
|
|
r_cons_printf ("\nVe: Visual Eval help:\n\n"
|
|
" q - quit menu\n"
|
|
" j/k - down/up keys\n"
|
|
" h/b - go back\n"
|
|
" $ - same as ?$ - show values of vars\n"
|
|
" e/' ' - edit/toggle current variable\n"
|
|
" E - edit variable with 'cfg.editor' (vi?)\n"
|
|
" +/- - increase/decrease numeric value (* and /, too)\n"
|
|
" : - enter command\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case ':':
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw(0);
|
|
{
|
|
char *cmd = prompt (":> ", NULL);
|
|
r_core_cmd (core, cmd, 1);
|
|
free (cmd);
|
|
}
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
r_cons_any_key (NULL);
|
|
r_cons_clear00 ();
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
R_API void r_core_visual_mounts(RCore *core) {
|
|
RList *list = NULL;
|
|
RFSRoot *fsroot = NULL;
|
|
RListIter *iter;
|
|
RFSFile *file;
|
|
RFSPartition *part;
|
|
int i, ch, option, mode, partition, dir, delta = 7;
|
|
char *str, path[4096], buf[1024], *root = NULL;
|
|
const char *n, *p;
|
|
|
|
dir = partition = option = mode = 0;
|
|
for (;;) {
|
|
/* Clear */
|
|
r_cons_clear00 ();
|
|
|
|
/* Show */
|
|
if (mode == 0) {
|
|
if (list) {
|
|
r_list_free (list);
|
|
list = NULL;
|
|
}
|
|
r_cons_printf ("Press '/' to navigate the root filesystem.\nPartitions:\n\n");
|
|
n = r_fs_partition_type_get (partition);
|
|
list = r_fs_partitions (core->fs, n, 0);
|
|
i = 0;
|
|
if (list) {
|
|
r_list_foreach (list, iter, part) {
|
|
if ((option-delta <= i) && (i <= option+delta)) {
|
|
if (option == i)
|
|
r_cons_printf (" > ");
|
|
else r_cons_printf (" ");
|
|
r_cons_printf ("%d %02x 0x%010"PFMT64x" 0x%010"PFMT64x"\n",
|
|
part->number, part->type,
|
|
part->start, part->start+part->length);
|
|
}
|
|
i++;
|
|
}
|
|
r_list_free (list);
|
|
list = NULL;
|
|
} else r_cons_printf ("Cannot read partition\n");
|
|
} else if (mode == 1) {
|
|
r_cons_printf ("Types:\n\n");
|
|
for(i=0;;i++) {
|
|
n = r_fs_partition_type_get (i);
|
|
if (!n) break;
|
|
r_cons_printf ("%s%s\n", (i==partition)?" > ":" ", n);
|
|
}
|
|
} else if (mode == 3) {
|
|
i = 0;
|
|
r_cons_printf ("Mountpoints:\n\n");
|
|
r_list_foreach (core->fs->roots, iter, fsroot) {
|
|
if (fsroot && (option-delta <= i) && (i <= option+delta)) {
|
|
r_cons_printf ("%s %s\n", (option == i)?" > ":" ",
|
|
fsroot->path);
|
|
}
|
|
i++;
|
|
}
|
|
} else {
|
|
if (root) {
|
|
list = r_fs_dir (core->fs, path);
|
|
if (list) {
|
|
r_cons_printf ("%s:\n\n", path);
|
|
i = 0;
|
|
r_list_foreach (list, iter, file) {
|
|
if ((dir-delta <= i) && (i <= dir+delta)) {
|
|
r_cons_printf ("%s%c %s\n", (dir == i)?" > ":" ",
|
|
file->type, file->name);
|
|
}
|
|
i++;
|
|
}
|
|
r_cons_printf ("\n");
|
|
r_list_free (list);
|
|
list = NULL;
|
|
} else r_cons_printf ("Cannot open '%s' directory\n", root);
|
|
} else r_cons_printf ("Root undefined\n");
|
|
}
|
|
if (mode==2) {
|
|
r_str_chop_path (path);
|
|
str = path + strlen (path);
|
|
strncat (path, "/", sizeof (path)-strlen (path)-1);
|
|
list = r_fs_dir (core->fs, path);
|
|
file = r_list_get_n (list, dir);
|
|
if (file && file->type != 'd')
|
|
r_core_cmdf (core, "px @ 0x%"PFMT64x"!64", file->off);
|
|
r_list_free (list);
|
|
list = NULL;
|
|
*str='\0';
|
|
}
|
|
r_cons_flush ();
|
|
|
|
/* Ask for option */
|
|
ch = r_cons_readchar ();
|
|
if (ch==-1||ch==4){
|
|
free (root);
|
|
return;
|
|
}
|
|
ch = r_cons_arrow_to_hjkl (ch);
|
|
switch (ch) {
|
|
case '/':
|
|
if (root) R_FREE (root);
|
|
root = strdup ("/");
|
|
strncpy (path, root, sizeof (path)-1);
|
|
R_FREE (root);
|
|
mode = 2;
|
|
break;
|
|
case 'l':
|
|
case '\r':
|
|
case '\n':
|
|
if (mode == 0) {
|
|
n = r_fs_partition_type_get (partition);
|
|
list = r_fs_partitions (core->fs, n, 0);
|
|
if (!list) {
|
|
r_cons_printf ("Unknown partition\n");
|
|
r_cons_any_key (NULL);
|
|
r_cons_flush ();
|
|
break;
|
|
}
|
|
part = r_list_get_n (list, option);
|
|
if (!part) {
|
|
r_cons_printf ("Unknown partition\n");
|
|
r_cons_any_key (NULL);
|
|
r_cons_flush ();
|
|
break;
|
|
}
|
|
p = r_fs_partition_type (n, part->type);
|
|
if (p) {
|
|
if (r_fs_mount (core->fs, p, "/root", part->start)) {
|
|
free (root);
|
|
root = strdup ("/root");
|
|
strncpy (path, root, sizeof (path)-1);
|
|
mode = 2;
|
|
} else {
|
|
r_cons_printf ("Cannot mount partition\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
}
|
|
} else {
|
|
r_cons_printf ("Unknown partition type\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
}
|
|
r_list_free (list);
|
|
list = NULL;
|
|
} else if (mode == 2){
|
|
r_str_chop_path (path);
|
|
strncat (path, "/", sizeof (path)-strlen (path)-1);
|
|
list = r_fs_dir (core->fs, path);
|
|
file = r_list_get_n (list, dir);
|
|
if (file) {
|
|
if (file->type == 'd') {
|
|
strncat (path, file->name, sizeof (path)-strlen (path)-1);
|
|
r_str_chop_path (path);
|
|
if (root && strncmp (root, path, strlen (root)-1))
|
|
strncpy (path, root, sizeof (path)-1);
|
|
} else {
|
|
r_core_cmdf (core, "s 0x%"PFMT64x, file->off);
|
|
r_fs_umount (core->fs, root);
|
|
free (root);
|
|
return;
|
|
}
|
|
} else {
|
|
r_cons_printf ("Unknown file\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
}
|
|
|
|
} else if (mode == 3) {
|
|
fsroot = r_list_get_n (core->fs->roots, option);
|
|
if (fsroot) {
|
|
root = strdup (fsroot->path);
|
|
strncpy (path, root, sizeof (path)-1);
|
|
}
|
|
mode = 2;
|
|
}
|
|
dir = partition = option = 0;
|
|
break;
|
|
case 'k':
|
|
if (mode == 0 || mode == 3) {
|
|
if (option > 0)
|
|
option--;
|
|
} else if (mode == 1) {
|
|
if (partition > 0)
|
|
partition--;
|
|
} else {
|
|
if (dir>0)
|
|
dir--;
|
|
}
|
|
break;
|
|
case 'j':
|
|
if (mode == 0) {
|
|
n = r_fs_partition_type_get (partition);
|
|
list = r_fs_partitions (core->fs, n, 0);
|
|
if (option < r_list_length (list)-1)
|
|
option++;
|
|
} else if (mode == 1) {
|
|
if (partition < r_fs_partition_get_size ()-1)
|
|
partition++;
|
|
} else if (mode == 3) {
|
|
if (option < r_list_length (core->fs->roots)-1)
|
|
option++;
|
|
} else {
|
|
list = r_fs_dir (core->fs, path);
|
|
if (dir < r_list_length (list)-1)
|
|
dir++;
|
|
}
|
|
break;
|
|
case 't':
|
|
mode = 1;
|
|
break;
|
|
case 'h':
|
|
if (mode == 2) {
|
|
if (!root) {
|
|
mode = 0;
|
|
} else
|
|
if (strcmp (path, root)) {
|
|
strcat (path, "/..");
|
|
r_str_chop_path (path);
|
|
} else {
|
|
r_fs_umount (core->fs, root);
|
|
mode = 0;
|
|
}
|
|
} else if (mode == 1) {
|
|
mode = 0;
|
|
} else {
|
|
return;
|
|
}
|
|
break;
|
|
case 'q':
|
|
if (mode == 2 && root) {
|
|
r_fs_umount (core->fs, root);
|
|
mode = 0;
|
|
} else {
|
|
return;
|
|
}
|
|
break;
|
|
case 'g':
|
|
if (mode == 2) {
|
|
r_str_chop_path (path);
|
|
str = path + strlen (path);
|
|
strncat (path, "/", sizeof (path)-strlen (path)-1);
|
|
list = r_fs_dir (core->fs, path);
|
|
file = r_list_get_n (list, dir);
|
|
if (file && root) {
|
|
strncat (path, file->name, sizeof (path)-strlen (path)-1);
|
|
r_str_chop_path (path);
|
|
if (strncmp (root, path, strlen (root)-1))
|
|
strncpy (path, root, sizeof (path)-1);
|
|
file = r_fs_open (core->fs, path);
|
|
if (file) {
|
|
r_fs_read (core->fs, file, 0, file->size);
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
r_line_set_prompt ("Dump path (ej: /tmp/file): ");
|
|
r_cons_fgets (buf, sizeof (buf)-1, 0, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
r_file_dump (buf, file->data, file->size, 0);
|
|
r_fs_close (core->fs, file);
|
|
r_cons_printf ("Done\n");
|
|
} else r_cons_printf ("Cannot dump file\n");
|
|
} else r_cons_printf ("Cannot dump file\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
*str='\0';
|
|
}
|
|
break;
|
|
case 'm':
|
|
mode = 3;
|
|
option = 0;
|
|
break;
|
|
case '?':
|
|
r_cons_clear00 ();
|
|
r_cons_printf ("\nVM: Visual Mount points help:\n\n");
|
|
r_cons_printf (" q - go back or quit menu\n");
|
|
r_cons_printf (" j/k - down/up keys\n");
|
|
r_cons_printf (" h/l - forward/go keys\n");
|
|
r_cons_printf (" t - choose partition type\n");
|
|
r_cons_printf (" g - dump file\n");
|
|
r_cons_printf (" m - show mountpoints\n");
|
|
r_cons_printf (" : - enter command\n");
|
|
r_cons_printf (" ? - show this help\n");
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case ':':
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (0);
|
|
r_line_set_prompt (":> ");
|
|
r_cons_fgets (buf, sizeof (buf)-1, 0, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
r_core_cmd (core, buf, 1);
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
static void var_index_show(RAnal *anal, RAnalFunction *fcn, ut64 addr, int idx) {
|
|
int i = 0;
|
|
RAnalVar *v;
|
|
RAnalVarAccess *x;
|
|
RListIter *iter, *iter2;
|
|
int window ;
|
|
|
|
// Adjust the windows size automaticaly
|
|
(void)r_cons_get_size (&window);
|
|
window-=5; // Size of printed things
|
|
|
|
int wdelta = (idx>5)?idx-5:0;
|
|
if (!fcn) return;
|
|
r_list_foreach(fcn->vars, iter, v) {
|
|
if (addr == 0 || (addr >= v->addr && addr <= v->eaddr)) {
|
|
if (i>=wdelta) {
|
|
if (i>window+wdelta) {
|
|
r_cons_printf("...\n");
|
|
break;
|
|
}
|
|
if (idx == i) r_cons_printf (" * ");
|
|
else r_cons_printf (" ");
|
|
#if 0
|
|
if (v->type->type == R_ANAL_TYPE_ARRAY) {
|
|
eprintf ("TODO: support for arrays\n");
|
|
r_cons_printf ("0x%08llx - 0x%08llx scope=%s type=%s name=%s delta=%d array=%d\n",
|
|
v->addr, v->eaddr, r_anal_var_scope_to_str (anal, v->scope),
|
|
r_anal_type_to_str (anal, v->type, ""),
|
|
v->name, v->delta, v->type->custom.a->count);
|
|
} else
|
|
#endif
|
|
{
|
|
char *s = r_anal_type_to_str (anal, v->type);
|
|
if (!s) s = strdup ("<unk>");
|
|
r_cons_printf ("0x%08llx - 0x%08llx scope=%d type=%s name=%s delta=%d\n",
|
|
v->addr, v->eaddr, v->scope, s, v->name, v->delta);
|
|
free (s);
|
|
}
|
|
r_list_foreach (v->accesses, iter2, x) {
|
|
r_cons_printf (" 0x%08llx %s\n", x->addr, x->set?"set":"get");
|
|
}
|
|
}
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// helper
|
|
static void function_rename(RCore *core, ut64 addr, const char *name) {
|
|
RListIter *iter;
|
|
RAnalFunction *fcn;
|
|
|
|
r_list_foreach (core->anal->fcns, iter, fcn) {
|
|
if (fcn->addr == addr) {
|
|
r_flag_unset_name (core->flags, fcn->name);
|
|
free (fcn->name);
|
|
fcn->name = strdup (name);
|
|
r_flag_set (core->flags, name, addr, r_anal_fcn_size (fcn));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// In visual mode, display function list
|
|
static ut64 var_functions_show(RCore *core, int idx, int show) {
|
|
int wdelta = (idx > 5)? idx - 5: 0;
|
|
ut64 seek = core->offset;
|
|
ut64 addr = core->offset;
|
|
RAnalFunction *fcn;
|
|
int window, i = 0;
|
|
RListIter *iter;
|
|
|
|
// Adjust the windows size automaticaly
|
|
(void)r_cons_get_size (&window);
|
|
window -= 8; // Size of printed things
|
|
|
|
r_list_foreach (core->anal->fcns, iter, fcn) {
|
|
if (i >= wdelta) {
|
|
if (i> window+wdelta) {
|
|
r_cons_printf ("...\n");
|
|
break;
|
|
}
|
|
if (idx == i) {
|
|
addr = fcn->addr;
|
|
}
|
|
if (show)
|
|
r_cons_printf ("%c%c 0x%08"PFMT64x" %4d %s\n",
|
|
(seek == fcn->addr)?'>':' ',
|
|
(idx==i)?'*':' ',
|
|
fcn->addr, r_anal_fcn_realsize (fcn), fcn->name);
|
|
}
|
|
i++;
|
|
}
|
|
return addr;
|
|
}
|
|
|
|
// In visual mode, display the variables.
|
|
static ut64 var_variables_show(RCore* core, int idx, int show) {
|
|
int i = 0;
|
|
const ut64 addr = var_functions_show (core, idx, 0);
|
|
RAnalFunction* fcn = r_anal_get_fcn_in (core->anal, addr, R_ANAL_FCN_TYPE_NULL);
|
|
int window;
|
|
int wdelta = (idx > 5) ? idx - 5 : 0;
|
|
RListIter *iter;
|
|
// arguments.
|
|
RList* list2 = r_anal_var_list (core->anal, fcn, 'v');
|
|
// variables.
|
|
RList* list = r_anal_var_list (core->anal, fcn, 'a');
|
|
r_list_join (list, list2);
|
|
RAnalVar* var;
|
|
// Adjust the window size automatically.
|
|
(void)r_cons_get_size (&window);
|
|
window -= 8; // Size of printed things.
|
|
|
|
// A new line so this looks reasonable.
|
|
r_cons_newline ();
|
|
|
|
r_list_foreach (list, iter, var) {
|
|
if (i >= wdelta) {
|
|
if (i > window + wdelta) {
|
|
r_cons_printf ("...\n");
|
|
break;
|
|
}
|
|
if (show) {
|
|
r_cons_printf ("%s %s %s @ %s%s0x%x\n",
|
|
var->kind=='v'?"var":"arg",
|
|
var->type, var->name,
|
|
core->anal->reg->name[R_REG_NAME_BP],
|
|
(var->kind=='v')?"-":"+",
|
|
var->delta);
|
|
}
|
|
}
|
|
++i;
|
|
}
|
|
r_list_free (list);
|
|
r_list_free (list2);
|
|
return addr;
|
|
}
|
|
|
|
static int level = 0;
|
|
static ut64 addr = 0;
|
|
static int option = 0;
|
|
static int printMode = 0;
|
|
#define lastPrintMode 5
|
|
|
|
static void r_core_visual_anal_refresh_column (RCore *core, int colpos) {
|
|
const ut64 addr = (level != 0 && level != 1)
|
|
? core->offset
|
|
: var_functions_show (core, option, 0);
|
|
RAnalFunction* fcn = r_anal_get_fcn_in(core->anal, addr, R_ANAL_FCN_TYPE_NULL);
|
|
int h, sz = 16, w = r_cons_get_size (&h);
|
|
if (fcn) sz = R_MIN (r_anal_fcn_size (fcn), h * 15); // max instr is 15 bytes.
|
|
|
|
const char *cmd, *printCmds[lastPrintMode] = {
|
|
"pdf", "afi", "pds", "pdc", "pdr"
|
|
};
|
|
if (printMode > 0 && printMode < lastPrintMode) {
|
|
cmd = printCmds[printMode];
|
|
} else {
|
|
cmd = printCmds[printMode = 0];
|
|
}
|
|
char *cmdf = r_str_newf ("%s @ 0x%"PFMT64x, cmd, addr);
|
|
if (!cmdf) {
|
|
return;
|
|
}
|
|
char *output = r_core_cmd_str (core, cmdf);
|
|
if (output) {
|
|
// 'h - 2' because we have two new lines in r_cons_printf
|
|
char *out = r_str_ansi_crop (output, 0, 0, w - colpos, h - 2);
|
|
r_cons_printf ("Visual code review (%s)\n%s\n", cmd, out);
|
|
free (out);
|
|
R_FREE (output);
|
|
}
|
|
free (cmdf);
|
|
}
|
|
|
|
static ut64 r_core_visual_anal_refresh (RCore *core) {
|
|
ut64 addr;
|
|
char old[1024];
|
|
int cols = r_cons_get_size (NULL);
|
|
if (!core) {
|
|
return 0LL;
|
|
}
|
|
old[0] = '\0';
|
|
addr = core->offset;
|
|
cols -= 50;
|
|
if (cols > 60) {
|
|
cols = 60;
|
|
}
|
|
|
|
r_cons_clear00 ();
|
|
r_cons_flush ();
|
|
r_core_visual_anal_refresh_column (core, cols);
|
|
if (cols > 30) {
|
|
r_cons_column (cols);
|
|
}
|
|
switch (level) {
|
|
// Show functions list help in visual mode
|
|
case 0:
|
|
r_cons_printf ("-[ functions ]---------------- \n"
|
|
"(a) add (x)xrefs (q)quit \n"
|
|
"(r) rename (c)calls (g)go \n"
|
|
"(d) delete (v)variables (?)help \n");
|
|
addr = var_functions_show (core, option, 1);
|
|
break;
|
|
case 1:
|
|
r_cons_printf (
|
|
"-[ variables ]----- 0x%08"PFMT64x"\n"
|
|
"(a) add (x)xrefs \n"
|
|
"(r) rename (g)go \n"
|
|
"(d) delete (q)quit \n", addr);
|
|
addr = var_variables_show (core, option, 1);
|
|
// var_index_show (core->anal, fcn, addr, option);
|
|
break;
|
|
case 2:
|
|
r_cons_printf ("Press 'q' to quit call refs\n");
|
|
r_cons_printf ("-[ calls ]----------------------- 0x%08"PFMT64x" (TODO)\n", addr);
|
|
// TODO: filter only the callrefs. but we cant grep here
|
|
sprintf(old, "afi @ 0x%08"PFMT64x, addr);
|
|
r_core_cmd0 (core, old);
|
|
break;
|
|
case 3:
|
|
r_cons_printf ("Press 'q' to view call refs\n");
|
|
r_cons_printf ("-[ xrefs ]----------------------- 0x%08"PFMT64x"\n", addr);
|
|
//sprintf (old, "axl~0x%08"PFMT64x, addr);
|
|
r_core_cmd0 (core, "pd 1");
|
|
//r_core_cmd0 (core, old);
|
|
break;
|
|
}
|
|
r_cons_flush ();
|
|
return addr;
|
|
}
|
|
|
|
/* Like emenu but for real */
|
|
R_API void r_core_visual_anal(RCore *core) {
|
|
char old[218];
|
|
int ch, _option = 0;
|
|
int nfcns = r_list_length (core->anal->fcns);
|
|
RConsEvent olde = core->cons->event_resize;
|
|
core->cons->event_resize = (RConsEvent) r_core_visual_anal_refresh;
|
|
level = 0;
|
|
addr = core->offset;
|
|
|
|
int asmbytes = r_config_get_i (core->config, "asm.bytes");
|
|
r_config_set_i (core->config, "asm.bytes", 0);
|
|
for (;;) {
|
|
addr = r_core_visual_anal_refresh (core);
|
|
ch = r_cons_readchar ();
|
|
if (ch == 4 || ch == -1) {
|
|
if (level == 0) {
|
|
goto beach;
|
|
}
|
|
level--;
|
|
continue;
|
|
}
|
|
ch = r_cons_arrow_to_hjkl (ch); // get ESC+char, return 'hjkl' char
|
|
switch (ch) {
|
|
case '?':
|
|
r_cons_clear ();
|
|
r_cons_printf (
|
|
"Usage: Vv [\n"
|
|
"Actions supported:\n"
|
|
" functions: Add, Modify, Delete, Xrefs Calls Vars\n"
|
|
" variables: Add, Modify, Delete\n"
|
|
"Moving:\n"
|
|
" j,k select next/prev item\n"
|
|
" J,K scroll next/prev page\n"
|
|
" h,q go back, quit\n"
|
|
" p,P switch next/prev print mode\n"
|
|
" l,ret enter, function\n"
|
|
);
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case ':':
|
|
r_core_visual_prompt (core);
|
|
r_cons_any_key (NULL);
|
|
continue;
|
|
case 'a':
|
|
switch (level) {
|
|
case 0:
|
|
eprintf ("TODO: Add new function manually\n");
|
|
/*
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (false);
|
|
r_line_set_prompt ("Address: ");
|
|
if (!r_cons_fgets (old, sizeof (old), 0, NULL)) break;
|
|
old[strlen (old)-1] = 0;
|
|
if (!*old) break;
|
|
addr = r_num_math (core->num, old);
|
|
r_line_set_prompt ("Size: ");
|
|
if (!r_cons_fgets (old, sizeof (old), 0, NULL)) break;
|
|
old[strlen (old)-1] = 0;
|
|
if (!*old) break;
|
|
size = r_num_math (core->num, old);
|
|
r_line_set_prompt ("Name: ");
|
|
if (!r_cons_fgets (old, sizeof (old), 0, NULL)) break;
|
|
old[strlen (old)-1] = 0;
|
|
r_flag_set (core->flags, old, addr, 0, 0);
|
|
//XXX sprintf(cmd, "CF %lld @ 0x%08llx", size, addr);
|
|
// XXX r_core_cmd0(core, cmd);
|
|
r_cons_set_raw (true);
|
|
r_cons_show_cursor (false);
|
|
*/
|
|
break;
|
|
case 1:
|
|
break;
|
|
}
|
|
break;
|
|
case 'r':
|
|
r_cons_show_cursor (true);
|
|
r_cons_set_raw (false);
|
|
r_line_set_prompt ("New name: ");
|
|
if (!r_cons_fgets (old, sizeof (old), 0, NULL)) break;
|
|
//old[strlen (old)-1] = 0;
|
|
function_rename (core, addr, old);
|
|
r_cons_set_raw (true);
|
|
r_cons_show_cursor (false);
|
|
break;
|
|
case 'p':
|
|
printMode ++;
|
|
break;
|
|
case 'P':
|
|
if (printMode == 0) {
|
|
printMode = lastPrintMode;
|
|
} else {
|
|
printMode --;
|
|
}
|
|
break;
|
|
case 'd':
|
|
switch (level) {
|
|
case 0:
|
|
eprintf ("TODO\n");
|
|
//data_del(addr, DATA_FUN, 0);
|
|
// XXX correcly remove all the data contained inside the size of the function
|
|
//flag_remove_at(addr);
|
|
break;
|
|
}
|
|
break;
|
|
case 'x': level = 3; break;
|
|
case 'c': level = 2; break;
|
|
case 'v': level = 1; break;
|
|
case 'j':
|
|
option++;
|
|
if (option >= nfcns) --option;
|
|
break;
|
|
case 'k':
|
|
option = (option<=0)? 0: option-1;
|
|
break;
|
|
case 'J':
|
|
{
|
|
int rows = 0;
|
|
r_cons_get_size (&rows);
|
|
option += (rows - 5);
|
|
if (option >= nfcns) {
|
|
option = nfcns - 1;
|
|
}
|
|
}
|
|
break;
|
|
case 'K':
|
|
{
|
|
int rows = 0;
|
|
r_cons_get_size (&rows);
|
|
option -= (rows - 5);
|
|
if (option < 0) {
|
|
option = 0;
|
|
}
|
|
}
|
|
break;
|
|
case 'g':
|
|
r_core_seek (core, addr, SEEK_SET);
|
|
goto beach;
|
|
case ' ':
|
|
case 'l':
|
|
level = 1;
|
|
_option = option;
|
|
break;
|
|
case 'h':
|
|
case 'b': // back
|
|
level = 0;
|
|
option = _option;
|
|
break;
|
|
case 'q':
|
|
if (level == 0) {
|
|
goto beach;
|
|
}
|
|
level--;
|
|
break;
|
|
}
|
|
}
|
|
beach:
|
|
core->cons->event_resize = olde;
|
|
level = 0;
|
|
r_config_set_i (core->config, "asm.bytes", asmbytes);
|
|
}
|
|
|
|
R_API void r_core_seek_next(RCore *core, const char *type) {
|
|
RListIter *iter;
|
|
ut64 next = UT64_MAX;
|
|
if (strstr (type, "opc")) {
|
|
RAnalOp aop;
|
|
if (r_anal_op (core->anal, &aop, core->offset, core->block, core->blocksize)) {
|
|
next = core->offset + aop.size;
|
|
} else {
|
|
eprintf ("Invalid opcode\n");
|
|
}
|
|
} else
|
|
if (strstr (type, "fun")) {
|
|
RAnalFunction *fcni;
|
|
r_list_foreach (core->anal->fcns, iter, fcni) {
|
|
if (fcni->addr < next && fcni->addr > core->offset) {
|
|
next = fcni->addr;
|
|
}
|
|
}
|
|
} else
|
|
if (strstr (type, "hit")) {
|
|
const char *pfx = r_config_get (core->config, "search.prefix");
|
|
RFlagItem *flag;
|
|
r_list_foreach (core->flags->flags, iter, flag) {
|
|
if (!strncmp (flag->name, pfx, strlen (pfx)))
|
|
if (flag->offset < next && flag->offset > core->offset)
|
|
next = flag->offset;
|
|
}
|
|
} else { // flags
|
|
RFlagItem *flag;
|
|
r_list_foreach (core->flags->flags, iter, flag) {
|
|
if (flag->offset < next && flag->offset > core->offset)
|
|
next = flag->offset;
|
|
}
|
|
}
|
|
if (next != UT64_MAX)
|
|
r_core_seek (core, next, 1);
|
|
}
|
|
|
|
R_API void r_core_seek_previous (RCore *core, const char *type) {
|
|
RListIter *iter;
|
|
ut64 next = 0;
|
|
if (strstr (type, "opc")) {
|
|
eprintf ("TODO: r_core_seek_previous (opc)\n");
|
|
} else
|
|
if (strstr (type, "fun")) {
|
|
RAnalFunction *fcni;
|
|
r_list_foreach (core->anal->fcns, iter, fcni) {
|
|
if (fcni->addr > next && fcni->addr < core->offset)
|
|
next = fcni->addr;
|
|
}
|
|
} else
|
|
if (strstr (type, "hit")) {
|
|
RFlagItem *flag;
|
|
const char *pfx = r_config_get (core->config, "search.prefix");
|
|
r_list_foreach (core->flags->flags, iter, flag) {
|
|
if (!strncmp (flag->name, pfx, strlen (pfx)))
|
|
if (flag->offset > next && flag->offset< core->offset)
|
|
next = flag->offset;
|
|
}
|
|
} else { // flags
|
|
RFlagItem *flag;
|
|
r_list_foreach (core->flags->flags, iter, flag) {
|
|
if (flag->offset > next && flag->offset < core->offset)
|
|
next = flag->offset;
|
|
}
|
|
}
|
|
if (next!=0)
|
|
r_core_seek (core, next, 1);
|
|
}
|
|
|
|
//define the data at offset according to the type (byte, word...) n times
|
|
static void define_data_ntimes (RCore *core, ut64 off, int times, int type) {
|
|
int i = 0;
|
|
r_meta_cleanup (core->anal, off, off + core->blocksize);
|
|
if (times < 0) {
|
|
times = 1;
|
|
}
|
|
for (i = 0; i < times; i++, off += type) {
|
|
r_meta_add (core->anal, R_META_TYPE_DATA, off, off + type, "");
|
|
}
|
|
}
|
|
|
|
static bool isDisasmPrint(int mode) {
|
|
return (mode == 1 || mode == 2);
|
|
}
|
|
|
|
R_API void r_core_visual_define (RCore *core) {
|
|
int plen = core->blocksize;
|
|
ut64 off = core->offset;
|
|
int i, h = 0, n, ch, ntotal = 0;
|
|
ut8 *p = core->block;
|
|
int rep = -1;
|
|
char *name;
|
|
int delta = 0;
|
|
if (core->print->cur_enabled) {
|
|
int cur = core->print->cur;
|
|
if (core->print->ocur != -1) {
|
|
plen = R_ABS (core->print->cur- core->print->ocur)+1;
|
|
if (core->print->ocur<cur) {
|
|
cur = core->print->ocur;
|
|
}
|
|
}
|
|
off += cur;
|
|
p += cur;
|
|
}
|
|
(void) r_cons_get_size (&h);
|
|
h -= 19;
|
|
if (h < 0) {
|
|
h = 0;
|
|
r_cons_clear00 ();
|
|
} else {
|
|
r_cons_gotoxy (0, h);
|
|
}
|
|
const char *lines[] = { ""
|
|
,"[Vd]- Define current block as:"
|
|
," $ define flag size"
|
|
," 1 edit bits"
|
|
," b set as byte"
|
|
," B set as short word (2 bytes)"
|
|
," c set as code"
|
|
," C define flag color (fc)"
|
|
," d set as data"
|
|
," e end of function"
|
|
," f analyze function"
|
|
," F format"
|
|
," i immediate base (b(in), o(ct), d(ec), h(ex), s(tr))"
|
|
," j merge down (join this and next functions)"
|
|
," k merge up (join this and previous function)"
|
|
," h highlight word"
|
|
," m manpage for current call"
|
|
," n rename flag used at cursor"
|
|
," r rename function"
|
|
," R find references /r"
|
|
," s set string"
|
|
," S set strings in current block"
|
|
," u undefine metadata here"
|
|
," x find xrefs to current address (./r)"
|
|
," w set as 32bit word"
|
|
," W set as 64bit word"
|
|
," q quit menu"
|
|
, NULL};
|
|
for (i = 0; lines[i]; i++) {
|
|
r_cons_fill_line ();
|
|
r_cons_printf ("\r%s\n", lines[i]);
|
|
}
|
|
r_cons_flush ();
|
|
// get ESC+char, return 'hjkl' char
|
|
repeat:
|
|
ch = r_cons_arrow_to_hjkl (r_cons_readchar ());
|
|
|
|
switch (ch) {
|
|
case 'F':
|
|
{
|
|
char cmd[128];
|
|
r_cons_show_cursor (true);
|
|
r_core_cmd0 (core, "pf?");
|
|
r_cons_flush ();
|
|
r_line_set_prompt ("format: ");
|
|
strcpy (cmd, "Cf 0 ");
|
|
if (r_cons_fgets (cmd+5, sizeof (cmd)-6, 0, NULL) > 0) {
|
|
r_core_cmd (core, cmd, 0);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
}
|
|
break;
|
|
case '1':
|
|
edit_bits (core);
|
|
break;
|
|
case 'x':
|
|
r_core_cmd0 (core, "./r");
|
|
break;
|
|
case 'B':
|
|
define_data_ntimes (core, off, rep, R_WORD_DATA);
|
|
break;
|
|
case 'i':
|
|
{
|
|
char str[128];
|
|
r_cons_show_cursor (true);
|
|
r_line_set_prompt ("immbase: ");
|
|
if (r_cons_fgets (str, sizeof (str), 0, NULL) > 0) {
|
|
r_core_cmdf (core, "ahi %s @ 0x%"PFMT64x, str, off);
|
|
}
|
|
}
|
|
break;
|
|
case 'b':
|
|
define_data_ntimes (core, off, rep, R_BYTE_DATA);
|
|
break;
|
|
case 'm':
|
|
{
|
|
char *man = NULL;
|
|
/* check for manpage */
|
|
RAnalOp *op = r_core_anal_op (core, off);
|
|
if (op) {
|
|
if (op->jump != UT64_MAX) {
|
|
RFlagItem *item = r_flag_get_i (core->flags, op->jump);
|
|
if (item) {
|
|
const char *ptr = r_str_lchr (item->name, '.');
|
|
if (ptr)
|
|
man = strdup (ptr+1);
|
|
}
|
|
}
|
|
r_anal_op_free (op);
|
|
}
|
|
if (man) {
|
|
char *p = strstr (man, "INODE");
|
|
if (p) *p = 0;
|
|
r_cons_clear();
|
|
r_cons_flush();
|
|
r_sys_cmdf ("man %s", man);
|
|
free (man);
|
|
}
|
|
r_cons_any_key (NULL);
|
|
}
|
|
break;
|
|
case 'n':
|
|
{
|
|
RAnalOp op;
|
|
char *q = NULL;
|
|
ut64 tgt_addr = UT64_MAX;
|
|
if (!isDisasmPrint (core->printidx)) {
|
|
break;
|
|
}
|
|
// TODO: get the aligned instruction even if the cursor is in the middle of it.
|
|
r_anal_op (core->anal, &op, off,
|
|
core->block + off - core->offset, 32);
|
|
|
|
tgt_addr = op.jump != UT64_MAX ? op.jump : op.ptr;
|
|
if (op.var) {
|
|
// q = r_str_newf ("?i Rename variable %s to;afvn %s `?y`", op.var->name, op.var->name);
|
|
#if 1
|
|
RAnalFunction *fcn = r_anal_get_fcn_in (core->anal, off, 0);
|
|
if (fcn) {
|
|
RAnalVar *bar = r_anal_var_get_byname (core->anal, fcn, op.var->name);
|
|
if (!bar) {
|
|
bar = r_anal_var_get_byname (core->anal, fcn, op.var->name);
|
|
if (!bar) {
|
|
bar = r_anal_var_get_byname (core->anal, fcn, op.var->name);
|
|
}
|
|
}
|
|
if (bar) {
|
|
char *newname = r_cons_input (sdb_fmt (0, "New variable name for '%s': ", bar->name));
|
|
if (newname) {
|
|
if (*newname) {
|
|
r_anal_var_rename (core->anal, fcn->addr, bar->scope,
|
|
bar->kind, bar->name, newname);
|
|
}
|
|
free (newname);
|
|
}
|
|
} else {
|
|
eprintf ("Cannot find variable\n");
|
|
r_sys_sleep (1);
|
|
}
|
|
} else {
|
|
eprintf ("Cannot find function\n");
|
|
r_sys_sleep (1);
|
|
}
|
|
#endif
|
|
} else if (tgt_addr != UT64_MAX) {
|
|
RAnalFunction *fcn = r_anal_get_fcn_at (core->anal, tgt_addr, R_ANAL_FCN_TYPE_NULL);
|
|
RFlagItem *f = r_flag_get_i (core->flags, tgt_addr);
|
|
if (fcn) {
|
|
q = r_str_newf ("?i Rename function %s to;afn `?y` 0x%"PFMT64x,
|
|
fcn->name, tgt_addr);
|
|
} else if (f) {
|
|
q = r_str_newf ("?i Rename flag %s to;fr %s `?y`",
|
|
f->name, f->name);
|
|
} else {
|
|
q = r_str_newf ("?i Create flag at 0x%"PFMT64x" named;f `?y` @ 0x%"PFMT64x,
|
|
tgt_addr, tgt_addr);
|
|
}
|
|
}
|
|
|
|
if (q) {
|
|
r_core_cmd0 (core, q);
|
|
free (q);
|
|
}
|
|
r_anal_op_fini (&op);
|
|
break;
|
|
}
|
|
case 'C':
|
|
{
|
|
RFlagItem *item = r_flag_get_i (core->flags, off);
|
|
if (item) {
|
|
char cmd[128];
|
|
r_cons_show_cursor (true);
|
|
r_cons_flush ();
|
|
r_line_set_prompt ("color: ");
|
|
if (r_cons_fgets (cmd, sizeof (cmd)-1, 0, NULL) > 0) {
|
|
r_flag_color (core->flags, item, cmd);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
} else {
|
|
eprintf ("Sorry. No flag here\n");
|
|
r_cons_any_key (NULL);
|
|
}
|
|
}
|
|
break;
|
|
case '$':
|
|
{
|
|
RFlagItem *item = r_flag_get_i (core->flags, off);
|
|
if (item) {
|
|
char cmd[128];
|
|
r_cons_printf ("Current flag size is: %d\n", item->size);
|
|
r_cons_show_cursor (true);
|
|
r_cons_flush ();
|
|
r_line_set_prompt ("new size: ");
|
|
if (r_cons_fgets (cmd, sizeof (cmd)-1, 0, NULL) > 0) {
|
|
item->size = r_num_math (core->num, cmd);
|
|
r_cons_set_raw (1);
|
|
r_cons_show_cursor (false);
|
|
}
|
|
} else {
|
|
eprintf ("Sorry. No flag here\n");
|
|
r_cons_any_key (NULL);
|
|
}
|
|
}
|
|
break;
|
|
case 'w':
|
|
define_data_ntimes (core, off, rep, R_DWORD_DATA);
|
|
break;
|
|
case 'W':
|
|
define_data_ntimes (core, off, rep, R_QWORD_DATA);
|
|
break;
|
|
case 'e':
|
|
// set function size
|
|
{
|
|
RAnalFunction *fcn = r_anal_get_fcn_in (core->anal, off, 0);
|
|
if (!fcn) {
|
|
fcn = r_anal_get_fcn_in (core->anal, core->offset, 0);
|
|
}
|
|
if (fcn) {
|
|
RAnalOp op;
|
|
ut64 size;
|
|
if (r_anal_op (core->anal, &op, off, core->block+delta,
|
|
core->blocksize-delta)) {
|
|
size = off - fcn->addr + op.size;
|
|
r_anal_fcn_resize (fcn, size);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 'j':
|
|
r_core_cmdf (core, "afm $$+$F @0x%08"PFMT64x, off);
|
|
break;
|
|
case 'k':
|
|
eprintf ("TODO: merge up\n");
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case 'h': // "Vdh"
|
|
r_core_cmdf (core, "?i highlight;e scr.highlight=`?y` @ 0x%08"PFMT64x, off);
|
|
break;
|
|
case 'r': // "Vdr"
|
|
r_core_cmdf (core, "?i new function name;afn `?y` @ 0x%08"PFMT64x, off);
|
|
break;
|
|
case 'R': // "VdR"
|
|
eprintf ("Finding references to 0x%08"PFMT64x" ...\n", off);
|
|
r_core_cmdf (core, "./r 0x%08"PFMT64x" @ $S", off);
|
|
break;
|
|
case 'S':
|
|
do {
|
|
n = r_str_nlen ((const char*)p+ntotal, plen-ntotal)+1;
|
|
if (n<2) break;
|
|
if (p[ntotal + n - 1])
|
|
break; // Not a \0 terminated string
|
|
name = malloc (n+10);
|
|
strcpy (name, "str.");
|
|
memcpy (name+4, (const char *)p+ntotal, n);
|
|
name[4+n] = '\0';
|
|
r_meta_add (core->anal, R_META_TYPE_STRING,
|
|
off+ntotal, off+n+ntotal, (const char *)name+4);
|
|
r_name_filter (name, n+10);
|
|
r_flag_set (core->flags, name, off+ntotal, n);
|
|
free (name);
|
|
ntotal += n;
|
|
} while (ntotal<plen);
|
|
break;
|
|
case 's':
|
|
{
|
|
int i;
|
|
// TODO: r_core_cmd0 (core, "Cz");
|
|
if (core->print->ocur != -1)
|
|
n = plen;
|
|
else n = r_str_nlen ((const char*)p, plen)+1;
|
|
name = malloc (n+10);
|
|
strcpy (name, "str.");
|
|
memcpy (name+4, (const char *)p, n);
|
|
name[4+n] = '\0';
|
|
for (i = 0; i < n; i++)
|
|
if (!name[4+i])
|
|
name[4+i]='_';
|
|
r_meta_add (core->anal, R_META_TYPE_STRING, off, off+n, (const char *)name+4);
|
|
r_name_filter (name, n+10);
|
|
r_flag_set (core->flags, name, off, n);
|
|
free (name);
|
|
}
|
|
break;
|
|
case 'd': // TODO: check
|
|
r_meta_cleanup (core->anal, off, off+plen);
|
|
r_meta_add (core->anal, R_META_TYPE_DATA, off, off+plen, "");
|
|
break;
|
|
case 'c': // TODO: check
|
|
r_meta_cleanup (core->anal, off, off+plen);
|
|
r_meta_add (core->anal, R_META_TYPE_CODE, off, off+plen, "");
|
|
break;
|
|
case 'u':
|
|
r_core_anal_undefine (core, off);
|
|
break;
|
|
case 'f':
|
|
{
|
|
int funsize = 0;
|
|
RAnalFunction *fcn = r_anal_get_fcn_in (core->anal, core->offset, 0);
|
|
if (fcn) {
|
|
r_anal_fcn_resize (fcn, core->offset - fcn->addr);
|
|
}
|
|
//int depth = r_config_get_i (core->config, "anal.depth");
|
|
if (core->print->cur_enabled) {
|
|
if (core->print->ocur != -1) {
|
|
funsize = 1 + R_ABS (core->print->cur - core->print->ocur);
|
|
}
|
|
//depth = 0;
|
|
}
|
|
r_cons_break (NULL, NULL);
|
|
r_core_cmdf (core, "af @ 0x%08" PFMT64x, off); // required for thumb autodetection
|
|
//r_core_anal_fcn (core, off, UT64_MAX, R_ANAL_REF_TYPE_NULL, depth);
|
|
r_cons_break_end ();
|
|
if (funsize) {
|
|
RAnalFunction *f = r_anal_get_fcn_in (core->anal, off, -1);
|
|
r_anal_fcn_set_size (f, funsize);
|
|
}
|
|
}
|
|
break;
|
|
case 'q':
|
|
default:
|
|
if (ch >= '0' && ch <= '9') {
|
|
if (rep < 0) {
|
|
rep = 0;
|
|
}
|
|
rep = rep * 10 + atoi ((char *)&ch);
|
|
goto repeat;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
R_API void r_core_visual_colors(RCore *core) {
|
|
char color[32], cstr[32];
|
|
const char *k, *kol;
|
|
int ch, opt = 0, oopt = -1;
|
|
ut8 r, g, b;
|
|
|
|
r = g = b = 0;
|
|
kol = r_cons_pal_get_color (opt);
|
|
r_cons_rgb_parse (kol, &r, &g, &b, NULL);
|
|
for (;;) {
|
|
r_cons_clear ();
|
|
k = r_cons_pal_get_i (opt);
|
|
if (!k) {
|
|
opt = 0;
|
|
k = r_cons_pal_get_i (opt);
|
|
}
|
|
r_cons_gotoxy (0, 0);
|
|
r_cons_rgb_str (cstr, r, g, b, 0);
|
|
r&=0xf;
|
|
g&=0xf;
|
|
b&=0xf;
|
|
sprintf (color, "rgb:%x%x%x", r, g, b);
|
|
//r_cons_printf ("COLOR%s(%sXXX)"Color_RESET"\n", kol, kol?kol+1:"");
|
|
r_cons_printf ("# Colorscheme %d - Use '.' and ':' to randomize palette\n"
|
|
"# Press 'rRgGbB', 'jk' or 'q'\nec %s %s # %d (%s)\n",
|
|
opt, k, color, atoi (cstr+7), cstr+1);
|
|
r_core_cmdf (core, "ec %s %s", k, color);
|
|
r_core_cmd0 (core, "pd 25");
|
|
r_cons_flush ();
|
|
ch = r_cons_readchar ();
|
|
ch = r_cons_arrow_to_hjkl (ch);
|
|
switch (ch) {
|
|
#define CASE_RGB(x,X,y) \
|
|
case x:y--;if(y>0x7f)y=0;break;\
|
|
case X:y++;if(y>15)y=15;break;
|
|
CASE_RGB ('R','r',r);
|
|
CASE_RGB ('G','g',g);
|
|
CASE_RGB ('B','b',b);
|
|
case 'q': return;
|
|
case 'k': opt--; break;
|
|
case 'j': opt++; break;
|
|
case 'K': opt=0; break;
|
|
case 'J': opt=0; break; // XXX must go to end
|
|
case ':': r_cons_pal_random (); break;
|
|
case '.':
|
|
r = r_num_rand (0xf);
|
|
g = r_num_rand (0xf);
|
|
b = r_num_rand (0xf);
|
|
break;
|
|
}
|
|
if (opt != oopt) {
|
|
kol = r_cons_pal_get_color (opt);
|
|
r_cons_rgb_parse (kol, &r, &g, &b, NULL);
|
|
oopt = opt;
|
|
}
|
|
}
|
|
}
|