mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-27 08:12:44 +00:00
a6c23407b2
fix #2908
506 lines
11 KiB
C
506 lines
11 KiB
C
/* Copyright radare2 2014-2015 - Author: pancake */
|
|
|
|
// pls move the typedefs into roons and rename it -> RConsPanel
|
|
|
|
#include <r_core.h>
|
|
|
|
typedef struct {
|
|
int x;
|
|
int y;
|
|
int w;
|
|
int h;
|
|
int depth;
|
|
int type;
|
|
ut64 addr;
|
|
char *cmd;
|
|
char *text;
|
|
} Panel;
|
|
|
|
#define PANEL_TYPE_FRAME 0
|
|
#define PANEL_TYPE_DIALOG 1
|
|
#define PANEL_TYPE_FLOAT 2
|
|
|
|
static int COLW = 40;
|
|
static RCore *_core;
|
|
static int n_panels = 0;
|
|
static void reloadPanels(RCore *core) ;
|
|
static int menu_pos = 0;
|
|
#define LIMIT 256
|
|
struct {
|
|
int panels[LIMIT];
|
|
int size;
|
|
} ostack;
|
|
|
|
static RConsCanvas *can;
|
|
static Panel *panels = NULL;
|
|
static int callgraph = 0;
|
|
static int instep = 0;
|
|
|
|
static int menu_x = 0;
|
|
static int menu_y = 0;
|
|
|
|
static const char *menus[] = {
|
|
"File", "Edit", "View", "Tools", "Debug", "Analyze", "Help",
|
|
NULL
|
|
};
|
|
|
|
static const char *menus_File[] = {
|
|
"New", "Open", "Close", "Info", "Quit",
|
|
NULL
|
|
};
|
|
|
|
static const char *menus_Edit[] = {
|
|
"Copy", "Paste", "Insert", "Assemble", "Fill",
|
|
NULL
|
|
};
|
|
|
|
static const char *menus_View[] = {
|
|
"Hexdump", "Disassembly", "Entropy",
|
|
NULL
|
|
};
|
|
|
|
static const char *menus_Tools[] = {
|
|
"Assembler", "Calculator", "Shell",
|
|
NULL
|
|
};
|
|
|
|
static const char *menus_Debug[] = {
|
|
"Registers", "Breakpoints", "Watchpoints", "Maps",
|
|
"Step", "Step Over", "Continue", "Cont until.",
|
|
NULL
|
|
};
|
|
|
|
static const char *menus_Analyze[] = {
|
|
"Function", "Program", "Calls", "Options",
|
|
NULL
|
|
};
|
|
|
|
static const char *menus_Help[] = {
|
|
"Commands", "License", "About",
|
|
NULL
|
|
};
|
|
|
|
static const char **menus_sub[] = {
|
|
menus_File,
|
|
menus_Edit,
|
|
menus_View,
|
|
menus_Tools,
|
|
menus_Debug,
|
|
menus_Analyze,
|
|
menus_Help,
|
|
NULL
|
|
};
|
|
|
|
|
|
// TODO: handle mouse wheel
|
|
|
|
static int curnode = 0;
|
|
|
|
#define G(x,y) r_cons_canvas_gotoxy (can, x, y)
|
|
#define W(x) r_cons_canvas_write (can, x)
|
|
#define B(x,y,w,h) r_cons_canvas_box(can, x,y,w,h,NULL)
|
|
#define B1(x,y,w,h) r_cons_canvas_box(can, x,y,w,h,Color_BLUE)
|
|
#define B2(x,y,w,h) r_cons_canvas_box(can, x,y,w,h,Color_MAGENTA)
|
|
#define L(x,y,x2,y2) r_cons_canvas_line(can, x,y,x2,y2,0)
|
|
#define L1(x,y,x2,y2) r_cons_canvas_line(can, x,y,x2,y2,1)
|
|
#define L2(x,y,x2,y2) r_cons_canvas_line(can, x,y,x2,y2,2)
|
|
#define F(x,y,x2,y2,c) r_cons_canvas_fill(can, x,y,x2,y2,c,0)
|
|
|
|
static void Panel_print(RConsCanvas *can, Panel *n, int cur) {
|
|
char title[128];
|
|
int delta_x = 0;
|
|
int delta_y = 0;
|
|
|
|
if (!can)
|
|
return;
|
|
// clear
|
|
F(n->x, n->y, n->w, n->h, ' ');
|
|
if (n->type == PANEL_TYPE_FRAME) {
|
|
if (cur) {
|
|
//F (n->x,n->y, n->w, n->h, '.');
|
|
snprintf (title, sizeof (title)-1,
|
|
" [ %s ] ", n->text);
|
|
} else {
|
|
snprintf (title, sizeof (title)-1,
|
|
" %s ", ""); //n->text);
|
|
}
|
|
if (G (n->x+1, n->y+1))
|
|
W (title); // delta_x
|
|
}
|
|
(void)G (n->x+2+delta_x, n->y+2);
|
|
//if (
|
|
// TODO: only refresh if n->refresh is set
|
|
// TODO: temporary crop depending on out of screen offsets
|
|
if (n->cmd && *n->cmd) {
|
|
char *foo = r_core_cmd_strf (_core, "%s", n->cmd);
|
|
char *text = r_str_crop (foo,
|
|
delta_x, delta_y, n->w, n->h);
|
|
if (text) {
|
|
W (text);
|
|
free (text);
|
|
} else {
|
|
W (n->text);
|
|
}
|
|
free (foo);
|
|
} else {
|
|
char *text = r_str_crop (n->text,
|
|
delta_x, delta_y, n->w+5, n->h);
|
|
if (text) {
|
|
W (text);
|
|
free (text);
|
|
} else {
|
|
W (n->text);
|
|
}
|
|
}
|
|
if (G (n->x+1, n->y+1))
|
|
W (title);
|
|
// TODO: check if node is traced or not and hsow proper color
|
|
// This info must be stored inside Panel* from RCore*
|
|
if (cur) {
|
|
B1 (n->x, n->y, n->w, n->h);
|
|
} else {
|
|
B (n->x, n->y, n->w, n->h);
|
|
}
|
|
}
|
|
|
|
static void Layout_run(Panel *panels) {
|
|
int h, w = r_cons_get_size (&h);
|
|
int i, j;
|
|
int colpos = w-COLW;
|
|
|
|
can->sx = 0;
|
|
can->sy = 0;
|
|
|
|
n_panels = 0;
|
|
for (i=0; panels[i].text; i++) {
|
|
if (panels[i].type == PANEL_TYPE_FRAME)
|
|
n_panels++;
|
|
}
|
|
for (i=j=0; panels[i].text; i++) {
|
|
switch (panels[i].type) {
|
|
case PANEL_TYPE_FLOAT:
|
|
panels[i].w = r_str_bounds (
|
|
panels[i].text,
|
|
&panels[i].h);
|
|
panels[i].h += 4;
|
|
break;
|
|
case PANEL_TYPE_FRAME:
|
|
if (j == 0) {
|
|
panels[i].x = 0;
|
|
panels[i].y = 1;
|
|
if (panels[j+1].text) {
|
|
panels[i].w = colpos;
|
|
} else {
|
|
panels[i].w = w;
|
|
}
|
|
panels[i].h = h-1;
|
|
} else {
|
|
int ph = ((h-1)/(n_panels-1));
|
|
panels[i].x = colpos;
|
|
panels[i].y = 1 + (ph*(j-1));
|
|
panels[i].w = w-colpos;
|
|
if (panels[i].w<0)
|
|
panels[i].w = 0;
|
|
panels[i].h = ph;
|
|
if (!panels[i+1].text) {
|
|
panels[i].h = h - panels[i].y;
|
|
}
|
|
}
|
|
j++;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int bbPanels (RCore *core, Panel **n) {
|
|
int i;
|
|
Panel *panels = calloc (sizeof (Panel), LIMIT); //(r_list_length (fcn->bbs)+1));
|
|
if (!panels)
|
|
return 0;
|
|
i = 0;
|
|
|
|
panels[i].text = strdup ("");
|
|
panels[i].addr = core->offset;
|
|
panels[i].type = PANEL_TYPE_FLOAT;
|
|
menu_pos = i;
|
|
i++;
|
|
|
|
panels[i].text = strdup ("Disassembly");
|
|
panels[i].cmd = r_str_newf ("pd $r-2");
|
|
panels[i].addr = core->offset;
|
|
panels[i].type = PANEL_TYPE_FRAME;
|
|
i++;
|
|
|
|
panels[i].text = strdup ("Symbols");
|
|
panels[i].cmd = strdup ("isq");
|
|
panels[i].addr = core->offset;
|
|
panels[i].type = PANEL_TYPE_FRAME;
|
|
i++;
|
|
n_panels = 2;
|
|
|
|
free (*n);
|
|
*n = panels;
|
|
panels[i].text = NULL;
|
|
Layout_run (panels);
|
|
return i;
|
|
}
|
|
|
|
// damn singletons.. there should be only one screen and therefor
|
|
// only one visual instance of the graph view. refactoring this
|
|
// into a struct makes the code to reference pointers unnecesarily
|
|
// we can look for a non-global solution here in the future if
|
|
// necessary
|
|
static void r_core_panels_refresh (RCore *core) {
|
|
char title[128];
|
|
int i, j, h, w = r_cons_get_size (&h);
|
|
if (instep && core->io->debug) {
|
|
r_core_cmd0 (core, "sr pc");
|
|
}
|
|
r_cons_clear00 ();
|
|
if (!can) {
|
|
return;
|
|
}
|
|
r_cons_canvas_resize (can, w, h);
|
|
r_cons_canvas_clear (can);
|
|
if (panels) {
|
|
if (menu_y>0) {
|
|
panels[menu_pos].x = menu_x * 6;
|
|
} else {
|
|
panels[menu_pos].x = w;
|
|
}
|
|
panels[menu_pos].y = 1;
|
|
free (panels[menu_pos].text);
|
|
panels[menu_pos].text = malloc(1024); //r_str_newf ("%d", menu_y);
|
|
panels[menu_pos].text[0] = 0;
|
|
int maxsub = 0;
|
|
for (i=0; menus_sub[i]; i++) { maxsub = i; }
|
|
if (menu_x >= 0 && menu_x <maxsub && menus_sub[menu_x]) {
|
|
for (j = 0; menus_sub[menu_x][j]; j++) {
|
|
if (menu_y-1 == j) {
|
|
strcat (panels[menu_pos].text, "> ");
|
|
} else {
|
|
strcat (panels[menu_pos].text, " ");
|
|
}
|
|
strcat (panels[menu_pos].text,
|
|
menus_sub[menu_x][j]);
|
|
strcat (panels[menu_pos].text, " \n");
|
|
}
|
|
}
|
|
for (i=0; panels[i].text; i++) {
|
|
if (i != curnode) {
|
|
Panel_print (can, &panels[i], i==curnode);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (menu_y) {
|
|
curnode = menu_pos;
|
|
}
|
|
// redraw current node to make it appear on top
|
|
if (curnode >= 0) {
|
|
Panel_print (can, &panels[curnode], 1);
|
|
}
|
|
Panel_print (can, &panels[menu_pos], menu_y);
|
|
|
|
(void)G (-can->sx, -can->sy);
|
|
char str[128];
|
|
title[0] = 0;
|
|
for (i=0; menus[i]; i++) {
|
|
if (menu_x == i) {
|
|
snprintf (str, sizeof (title)-1, "[%s]", menus[i]);
|
|
} else {
|
|
snprintf (str, sizeof (title)-1, " %s ", menus[i]);
|
|
}
|
|
strcat (title, str);
|
|
}
|
|
W (title);
|
|
|
|
snprintf (title, sizeof (title)-1,
|
|
"[0x%08"PFMT64x"]", core->offset);
|
|
(void)G (-can->sx + w-strlen (title), -can->sy);
|
|
W (title);
|
|
|
|
r_cons_canvas_print (can);
|
|
r_cons_flush ();
|
|
}
|
|
|
|
static void reloadPanels(RCore *core) {
|
|
//W("HELLO WORLD");
|
|
Layout_run (panels);
|
|
}
|
|
|
|
R_API int r_core_visual_panels(RCore *core) {
|
|
#define OS_INIT() ostack.size = 0; ostack.panels[0] = 0;
|
|
#define OS_PUSH(x) if (ostack.size<LIMIT) {ostack.panels[++ostack.size]=x;}
|
|
#define OS_POP() ((ostack.size>0)? ostack.panels[--ostack.size]:0)
|
|
int okey, key, wheel;
|
|
int w, h;
|
|
int asm_comments = 0;
|
|
int asm_bytes = 0;
|
|
n_panels = 0;
|
|
panels = NULL;
|
|
callgraph = 0;
|
|
_core = core;
|
|
|
|
OS_INIT();
|
|
w = r_cons_get_size (&h);
|
|
can = r_cons_canvas_new (w, h);
|
|
can->linemode = 1;
|
|
can->color = r_config_get_i (core->config, "scr.color");
|
|
// disable colors in disasm because canvas doesnt supports ansi text yet
|
|
r_config_set_i (core->config, "scr.color", 0);
|
|
//can->color = 0;
|
|
if (!can) {
|
|
eprintf ("Cannot create RCons.canvas context\n");
|
|
return R_FALSE;
|
|
}
|
|
n_panels = bbPanels (core, &panels);
|
|
if (!panels) {
|
|
r_config_set_i (core->config, "scr.color", can->color);
|
|
free (can);
|
|
return R_FALSE;
|
|
}
|
|
|
|
reloadPanels (core);
|
|
|
|
asm_comments = r_config_get_i (core->config, "asm.comments");
|
|
r_config_set_i (core->config, "asm.comments", 0);
|
|
asm_bytes = r_config_get_i (core->config, "asm.bytes");
|
|
r_config_set_i (core->config, "asm.bytes", 0);
|
|
|
|
repeat:
|
|
core->cons->event_data = core;
|
|
core->cons->event_resize = \
|
|
(RConsEvent)r_core_panels_refresh;
|
|
w = r_cons_get_size (&h);
|
|
Layout_run (panels);
|
|
r_core_panels_refresh (core);
|
|
wheel = r_config_get_i (core->config, "scr.wheel");
|
|
if (wheel)
|
|
r_cons_enable_mouse (R_TRUE);
|
|
|
|
// r_core_graph_inputhandle()
|
|
okey = r_cons_readchar ();
|
|
key = r_cons_arrow_to_hjkl (okey);
|
|
|
|
switch (key) {
|
|
case ' ':
|
|
case '\n':
|
|
if (menu_y) {
|
|
const char *action = menus_sub[menu_x][menu_y-1];
|
|
eprintf ("ACTION %s\n", action);
|
|
if (strstr (action, "New")) {
|
|
int i;
|
|
// add new panel for testing
|
|
for (i=0; panels[i].text; i++) {
|
|
// find last panel
|
|
}
|
|
panels[i].text = strdup ("Test");
|
|
panels[i].cmd = r_str_newf ("pxW $r-2");
|
|
panels[i].addr = core->offset;
|
|
panels[i].type = PANEL_TYPE_FRAME;
|
|
i++;
|
|
n_panels++;
|
|
panels[i].text = NULL;
|
|
}
|
|
if (strstr (action, "Quit")) {
|
|
goto beach;
|
|
}
|
|
}
|
|
break;
|
|
case '?':
|
|
r_cons_clear00 ();
|
|
r_cons_printf ("Visual Ascii Art Panels:\n"
|
|
" ! run r2048 game\n"
|
|
" . - center graph to the current node\n"
|
|
" : - run r2 command in prompt\n"
|
|
" hl - toggle scr.color\n"
|
|
" HL - move vertical column split\n"
|
|
" JK - select prev/next panels\n"
|
|
" jk - scroll/select menu\n"
|
|
" q - quit, back to visual mode\n"
|
|
);
|
|
r_cons_flush ();
|
|
r_cons_any_key (NULL);
|
|
break;
|
|
case ':':
|
|
core->vmode = R_FALSE;
|
|
r_core_visual_prompt_input (core);
|
|
core->vmode = R_TRUE;
|
|
break;
|
|
case 'C':
|
|
can->color = !!!can->color; //WTF
|
|
//r_config_swap (core->config, "scr.color");
|
|
// refresh graph
|
|
// reloadPanels (core);
|
|
break;
|
|
case '!':
|
|
r_cons_2048 ();
|
|
break;
|
|
case 'j':
|
|
if (panels[curnode].type == PANEL_TYPE_FLOAT) {
|
|
if (menus_sub[menu_x][menu_y])
|
|
menu_y ++;
|
|
}
|
|
break;
|
|
case 'k':
|
|
if (panels[curnode].type == PANEL_TYPE_FLOAT) {
|
|
menu_y --;
|
|
if (menu_y<0)
|
|
menu_y = 0;
|
|
}
|
|
break;
|
|
case 'J':
|
|
curnode++;
|
|
if (!panels[curnode].text) {
|
|
curnode--;
|
|
}
|
|
break;
|
|
case 'K':
|
|
curnode--;
|
|
if (curnode<0)
|
|
curnode = 0;
|
|
break;
|
|
case 'H':
|
|
COLW += 4;
|
|
break;
|
|
case 'L':
|
|
COLW -= 4;
|
|
if (COLW<0)
|
|
COLW=0;
|
|
break;
|
|
case 'h':
|
|
if (menu_x) {
|
|
menu_x --;
|
|
menu_y = menu_y?1:0;
|
|
}
|
|
break;
|
|
case 'l':
|
|
if (menus[menu_x + 1]) {
|
|
menu_x ++;
|
|
menu_y = menu_y?1:0;
|
|
}
|
|
break;
|
|
case 'q':
|
|
case -1: // EOF
|
|
goto beach;
|
|
case 27: // ESC
|
|
if (r_cons_readchar () == 91) {
|
|
if (r_cons_readchar () == 90) {
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
//eprintf ("Key %d\n", key);
|
|
//sleep (1);
|
|
break;
|
|
}
|
|
goto repeat;
|
|
beach:
|
|
free (panels);
|
|
r_config_set_i (core->config, "scr.color", can->color);
|
|
free (can);
|
|
r_config_set_i (core->config, "asm.comments", asm_comments);
|
|
r_config_set_i (core->config, "asm.bytes", asm_bytes);
|
|
return R_TRUE;
|
|
}
|