2010-02-26 20:00:03 +00:00
|
|
|
|
/* radare - LGPL - Copyright 2009-2010 */
|
|
|
|
|
/* nibble<.ds@gmail.com> */
|
|
|
|
|
/* pancake<nopcode.org> */
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
|
|
#include "r_core.h"
|
2010-05-19 00:39:01 +00:00
|
|
|
|
#include "r_io.h"
|
2009-02-05 21:08:46 +00:00
|
|
|
|
#include "r_flags.h"
|
|
|
|
|
#include "r_hash.h"
|
|
|
|
|
#include "r_asm.h"
|
2009-02-26 14:15:19 +00:00
|
|
|
|
#include "r_anal.h"
|
2009-12-31 00:27:03 +00:00
|
|
|
|
#include "r_util.h"
|
2010-01-07 22:05:45 +00:00
|
|
|
|
#include "r_bp.h"
|
2009-02-26 14:15:19 +00:00
|
|
|
|
|
2009-12-31 00:27:03 +00:00
|
|
|
|
#include <sys/types.h>
|
2009-02-18 00:43:57 +00:00
|
|
|
|
#include <stdarg.h>
|
2010-04-14 20:28:45 +00:00
|
|
|
|
/* TODO: move to print/disasm.c */
|
|
|
|
|
static void r_print_disasm(RPrint *p, RCore *core, ut64 addr, ut8 *buf, int len, int l) {
|
2010-05-19 00:39:01 +00:00
|
|
|
|
RAnalFcn *fcni = NULL;
|
2010-04-14 20:28:45 +00:00
|
|
|
|
int ret, idx, i, j, k;
|
|
|
|
|
int middle = 0;
|
2010-05-19 01:55:20 +00:00
|
|
|
|
int stackptr = 0, ostackptr;
|
2010-04-14 20:28:45 +00:00
|
|
|
|
char str[128];
|
|
|
|
|
char *line;
|
|
|
|
|
char *comment;
|
|
|
|
|
char *opstr;
|
|
|
|
|
char *osl = NULL; // old source line
|
|
|
|
|
RAsmAop asmop;
|
|
|
|
|
RAnalAop analop;
|
|
|
|
|
RFlagItem *flag;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
int counter = 0;
|
2010-04-14 20:28:45 +00:00
|
|
|
|
|
|
|
|
|
// TODO: All those options must be print flags
|
2010-05-19 22:59:42 +00:00
|
|
|
|
int show_color = r_config_get_i (core->config, "scr.color");
|
|
|
|
|
int pseudo = r_config_get_i (core->config, "asm.pseudo");
|
|
|
|
|
int filter = r_config_get_i (core->config, "asm.filter");
|
|
|
|
|
int show_lines = r_config_get_i (core->config, "asm.lines");
|
|
|
|
|
int show_dwarf = r_config_get_i (core->config, "asm.dwarf");
|
|
|
|
|
int linesout = r_config_get_i (core->config, "asm.linesout");
|
|
|
|
|
int adistrick = r_config_get_i (core->config, "asm.middle"); // TODO: find better name
|
|
|
|
|
int show_offset = r_config_get_i (core->config, "asm.offset");
|
|
|
|
|
int show_bytes = r_config_get_i (core->config, "asm.bytes");
|
|
|
|
|
int show_comments = r_config_get_i (core->config, "asm.comments");
|
|
|
|
|
int show_stackptr = r_config_get_i (core->config, "asm.stackptr");
|
2010-04-14 20:28:45 +00:00
|
|
|
|
int linesopts = 0;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
int nb, nbytes = r_config_get_i (core->config, "asm.nbytes");
|
2010-04-14 20:28:45 +00:00
|
|
|
|
nb = nbytes*2;
|
|
|
|
|
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (r_config_get_i (core->config, "asm.linesstyle"))
|
2010-04-14 20:28:45 +00:00
|
|
|
|
linesopts |= R_ANAL_REFLINE_STYLE;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (r_config_get_i (core->config, "asm.lineswide"))
|
2010-04-14 20:28:45 +00:00
|
|
|
|
linesopts |= R_ANAL_REFLINE_WIDE;
|
|
|
|
|
|
2010-05-20 07:29:39 +00:00
|
|
|
|
// uhm... is this necesary? imho can be removed
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_asm_set_pc (core->assembler, core->offset);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
#if 0
|
|
|
|
|
/* find last function else stackptr=0 */
|
|
|
|
|
{
|
|
|
|
|
RAnalFcn *fcni;
|
|
|
|
|
RListIter *iter;
|
|
|
|
|
|
|
|
|
|
r_list_foreach (core->anal.fcns, iter, fcni) {
|
|
|
|
|
if (addr >= fcni->addr && addr<(fcni->addr+fcni->size)) {
|
|
|
|
|
stack_ptr = fcni->stack;
|
|
|
|
|
r_cons_printf ("/* function: %s (%d) */\n", fcni->name, fcni->size, stack_ptr);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2010-05-19 22:59:42 +00:00
|
|
|
|
// TODO: make anal->reflines implicit
|
|
|
|
|
free (core->reflines); // TODO: leak
|
|
|
|
|
core->reflines = r_anal_reflines_get (core->anal, core->offset,
|
2010-04-14 20:28:45 +00:00
|
|
|
|
buf, len, -1, linesout);
|
|
|
|
|
for (i=idx=ret=0; idx < len && i<l; idx+=ret,i++) {
|
|
|
|
|
ut64 addr = core->offset + idx;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_asm_set_pc (core->assembler, addr);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
if (show_comments) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
comment = r_meta_get_string (core->meta, R_META_COMMENT, addr);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
if (comment) {
|
|
|
|
|
r_cons_strcat (comment);
|
|
|
|
|
free (comment);
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
line = r_anal_reflines_str (core->anal, core->reflines, addr, linesopts);
|
|
|
|
|
ret = r_asm_disassemble (core->assembler, &asmop, buf+idx, len-idx);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
if (ret<1) {
|
|
|
|
|
ret = 1;
|
|
|
|
|
eprintf ("** invalid opcode at 0x%08"PFMT64x" **\n",
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->assembler->pc + ret);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_anal_aop (core->anal, &analop, addr, buf+idx, (int)(len-idx));
|
2010-04-14 20:28:45 +00:00
|
|
|
|
|
|
|
|
|
if (adistrick)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
middle = r_anal_reflines_middle (core->anal,
|
|
|
|
|
core->reflines, addr, analop.length);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
{
|
|
|
|
|
int found = 0;
|
|
|
|
|
RListIter *iter;
|
|
|
|
|
RAnalFcn *f = fcni;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_list_foreach (core->anal->fcns, iter, fcni) {
|
2010-05-19 00:39:01 +00:00
|
|
|
|
if (addr == fcni->addr) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cons_printf ("/* function: %s (%d) */\n",
|
|
|
|
|
fcni->name, fcni->size);
|
2010-05-19 01:55:20 +00:00
|
|
|
|
stackptr = 0;
|
2010-05-19 00:39:01 +00:00
|
|
|
|
found = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!found)
|
|
|
|
|
fcni = f;
|
|
|
|
|
}
|
|
|
|
|
if (fcni) {
|
|
|
|
|
if (addr >= fcni->addr+fcni->size-1) {
|
|
|
|
|
r_cons_printf ("\\*");
|
|
|
|
|
fcni = NULL;
|
|
|
|
|
} else
|
|
|
|
|
if (addr >= fcni->addr)
|
|
|
|
|
r_cons_printf (": ");
|
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
flag = r_flag_get_i (core->flags, core->offset+idx);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
if (flag && !show_bytes) {
|
|
|
|
|
if (show_lines && line)
|
|
|
|
|
r_cons_strcat (line);
|
|
|
|
|
if (show_offset) {
|
|
|
|
|
if (show_color)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cons_printf (Color_GREEN"0x%08"PFMT64x
|
|
|
|
|
" "Color_RESET, core->offset + idx);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
else r_cons_printf ("0x%08"PFMT64x" ", core->offset + idx);
|
|
|
|
|
}
|
|
|
|
|
r_cons_printf ("%s:\n", flag->name);
|
|
|
|
|
}
|
2010-04-14 20:28:45 +00:00
|
|
|
|
|
|
|
|
|
if (show_lines && line)
|
|
|
|
|
r_cons_strcat (line);
|
|
|
|
|
if (show_offset) {
|
|
|
|
|
if (show_color)
|
2010-05-19 00:39:01 +00:00
|
|
|
|
r_cons_printf (Color_GREEN"0x%08"PFMT64x
|
|
|
|
|
" "Color_RESET, core->offset + idx);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
else r_cons_printf ("0x%08"PFMT64x" ", core->offset + idx);
|
|
|
|
|
}
|
2010-05-18 17:38:47 +00:00
|
|
|
|
if (show_stackptr) {
|
2010-05-19 01:55:20 +00:00
|
|
|
|
if (analop.type == R_ANAL_OP_TYPE_RET)
|
|
|
|
|
stackptr = 0;
|
|
|
|
|
ostackptr = stackptr;
|
|
|
|
|
stackptr += analop.stackptr;
|
|
|
|
|
r_cons_printf ("%3d%s ", stackptr,
|
|
|
|
|
stackptr>ostackptr?"+":stackptr<ostackptr?"-":" ");
|
2010-05-18 17:38:47 +00:00
|
|
|
|
}
|
2010-05-19 00:39:01 +00:00
|
|
|
|
if (show_bytes) {
|
2010-04-14 20:28:45 +00:00
|
|
|
|
char pad[64];
|
|
|
|
|
char *str, *extra = " ";
|
|
|
|
|
if (!flag) {
|
|
|
|
|
str = strdup (asmop.buf_hex);
|
|
|
|
|
if (strlen (str) > nb) {
|
|
|
|
|
str[nb] = '.';
|
|
|
|
|
str[nb+1] = '\0';
|
|
|
|
|
extra = "";
|
|
|
|
|
}
|
|
|
|
|
k = nb-strlen (str);
|
|
|
|
|
if (k<0) k = 0;
|
|
|
|
|
for (j=0; j<k; j++)
|
|
|
|
|
pad[j] = ' ';
|
|
|
|
|
pad[j] = '\0';
|
|
|
|
|
if (show_color) {
|
|
|
|
|
char *nstr = r_print_hexpair (p, str);
|
|
|
|
|
free (str);
|
|
|
|
|
str = nstr;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
str = strdup (flag->name);
|
|
|
|
|
k = nb-strlen (str)-2;
|
|
|
|
|
if (k<0) k = 0;
|
|
|
|
|
for (j=0; j<k; j++)
|
|
|
|
|
pad[j] = ' ';
|
|
|
|
|
pad[j] = '\0';
|
|
|
|
|
}
|
|
|
|
|
if (flag) {
|
|
|
|
|
if (show_color)
|
|
|
|
|
r_cons_printf (Color_BWHITE"*[ %s%s] "Color_RESET, pad, str);
|
|
|
|
|
else r_cons_printf ("*[ %s%s] ", pad, str);
|
|
|
|
|
} else r_cons_printf (" %s %s %s", pad, str, extra);
|
|
|
|
|
free (str);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
}
|
2010-04-14 20:28:45 +00:00
|
|
|
|
if (show_color) {
|
|
|
|
|
switch (analop.type) {
|
|
|
|
|
case R_ANAL_OP_TYPE_NOP:
|
|
|
|
|
r_cons_printf (Color_BLUE);
|
|
|
|
|
break;
|
|
|
|
|
case R_ANAL_OP_TYPE_JMP:
|
|
|
|
|
case R_ANAL_OP_TYPE_CJMP:
|
2010-05-19 22:59:42 +00:00
|
|
|
|
case R_ANAL_OP_TYPE_UJMP:
|
2010-04-14 20:28:45 +00:00
|
|
|
|
r_cons_printf (Color_GREEN);
|
|
|
|
|
break;
|
|
|
|
|
case R_ANAL_OP_TYPE_CALL:
|
|
|
|
|
r_cons_printf (Color_BGREEN);
|
|
|
|
|
break;
|
|
|
|
|
case R_ANAL_OP_TYPE_SWI:
|
|
|
|
|
r_cons_printf (Color_MAGENTA);
|
|
|
|
|
break;
|
|
|
|
|
case R_ANAL_OP_TYPE_RET:
|
|
|
|
|
r_cons_printf (Color_RED);
|
|
|
|
|
break;
|
|
|
|
|
case R_ANAL_OP_TYPE_LOAD:
|
|
|
|
|
r_cons_printf (Color_YELLOW);
|
|
|
|
|
break;
|
|
|
|
|
case R_ANAL_OP_TYPE_STORE:
|
|
|
|
|
r_cons_printf (Color_BYELLOW);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (pseudo) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_parse_parse (core->parser, asmop.buf_asm, str);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
opstr = str;
|
|
|
|
|
} else if (filter) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_parse_filter (core->parser, core->flags, asmop.buf_asm, str);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
opstr = str;
|
|
|
|
|
} else opstr = asmop.buf_asm;
|
|
|
|
|
r_cons_strcat (opstr);
|
|
|
|
|
if (show_color)
|
|
|
|
|
r_cons_strcat (Color_RESET);
|
|
|
|
|
if (show_dwarf) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
char *sl = r_bin_meta_get_source_line (core->bin, addr);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
int len = strlen (opstr);
|
|
|
|
|
if (len<30)
|
|
|
|
|
len = 30-len;
|
|
|
|
|
if (sl)
|
|
|
|
|
if (!osl || (osl && strcmp (sl, osl))) {
|
|
|
|
|
while (len--)
|
|
|
|
|
r_cons_strcat (" ");
|
|
|
|
|
if (show_color)
|
|
|
|
|
r_cons_printf (Color_TURQOISE" ; %s"Color_RESET, sl);
|
|
|
|
|
else r_cons_printf (" ; %s\n", sl);
|
|
|
|
|
free (osl);
|
|
|
|
|
osl = sl;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (middle != 0) {
|
|
|
|
|
ret = ret-middle;
|
|
|
|
|
r_cons_printf (" ; *middle* %d", ret);
|
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
switch (analop.type) {
|
|
|
|
|
case R_ANAL_OP_TYPE_CALL:
|
|
|
|
|
case R_ANAL_OP_TYPE_JMP:
|
|
|
|
|
case R_ANAL_OP_TYPE_CJMP:
|
|
|
|
|
counter++;
|
|
|
|
|
if (counter>9)
|
|
|
|
|
r_cons_printf (" [?]");
|
|
|
|
|
else r_cons_printf (" [%d]", counter);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2010-04-14 20:28:45 +00:00
|
|
|
|
r_cons_newline ();
|
|
|
|
|
if (line) {
|
|
|
|
|
if (show_lines && analop.type == R_ANAL_OP_TYPE_RET) {
|
|
|
|
|
if (strchr (line, '>'))
|
|
|
|
|
memset (line, ' ', strlen (line));
|
|
|
|
|
r_cons_strcat (line);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
r_cons_strcat ("/* ------------ */\n");
|
2010-04-14 20:28:45 +00:00
|
|
|
|
}
|
|
|
|
|
free (line);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
free (osl);
|
|
|
|
|
}
|
|
|
|
|
/* </move> */
|
2009-02-18 00:43:57 +00:00
|
|
|
|
|
2009-09-02 00:10:51 +00:00
|
|
|
|
static int cmd_io_system(void *data, const char *input);
|
|
|
|
|
|
2010-04-08 10:29:47 +00:00
|
|
|
|
static int cmd_zign(void *data, const char *input) {
|
2010-04-08 14:20:41 +00:00
|
|
|
|
RAnalFcn *fcni;
|
|
|
|
|
RListIter *iter;
|
2010-04-08 13:48:53 +00:00
|
|
|
|
RSignItem *item;
|
2010-04-08 10:29:47 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-04-08 14:20:41 +00:00
|
|
|
|
char *ptr, *name;
|
|
|
|
|
int i, fd, len;
|
2010-04-08 10:29:47 +00:00
|
|
|
|
|
|
|
|
|
switch (input[0]) {
|
2010-04-08 14:20:41 +00:00
|
|
|
|
case 'g':
|
|
|
|
|
if (input[1]==' ' && input[2]) {
|
|
|
|
|
ptr = strchr (input+2, ' ');
|
|
|
|
|
if (ptr) {
|
|
|
|
|
*ptr = '\0';
|
|
|
|
|
fd = open (ptr+1, O_RDWR|O_CREAT, 0644);
|
|
|
|
|
if (fd == -1) {
|
|
|
|
|
eprintf ("Cannot open %s in read-write\n", ptr+1);
|
|
|
|
|
return R_FALSE;
|
|
|
|
|
}
|
|
|
|
|
r_cons_singleton ()->fdout = fd;
|
|
|
|
|
r_cons_strcat ("# Signatures\n");
|
|
|
|
|
}
|
|
|
|
|
r_cons_printf ("zp %s\n", input+2);
|
|
|
|
|
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_list_foreach (core->anal->fcns, iter, fcni) {
|
2010-04-08 14:20:41 +00:00
|
|
|
|
ut8 buf[128];
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (r_io_read_at (core->io, fcni->addr, buf, sizeof (buf)) == sizeof (buf)) {
|
|
|
|
|
RFlagItem *flag = r_flag_get_i (core->flags, fcni->addr);
|
2010-04-08 14:20:41 +00:00
|
|
|
|
if (flag) {
|
|
|
|
|
name = flag->name;
|
|
|
|
|
r_cons_printf ("zb %s ", name);
|
|
|
|
|
len = (fcni->size>sizeof (buf))?sizeof (buf):fcni->size;
|
|
|
|
|
for (i=0; i<len; i++) {
|
|
|
|
|
r_cons_printf ("%02x", buf[i]);
|
|
|
|
|
}
|
|
|
|
|
r_cons_newline ();
|
2010-04-14 11:02:23 +00:00
|
|
|
|
} else eprintf ("Unnamed function at 0x%08"PFMT64x"\n", fcni->addr);
|
|
|
|
|
} else eprintf ("Cannot read at 0x%08"PFMT64x"\n", fcni->addr);
|
2010-04-08 14:20:41 +00:00
|
|
|
|
}
|
|
|
|
|
if (ptr) {
|
|
|
|
|
r_cons_flush ();
|
|
|
|
|
r_cons_singleton ()->fdout = 1;
|
|
|
|
|
close (fd);
|
|
|
|
|
}
|
|
|
|
|
} else eprintf ("Usage: zg libc [libc.sig]\n");
|
|
|
|
|
break;
|
2010-04-08 13:48:53 +00:00
|
|
|
|
case 'p':
|
|
|
|
|
if (!input[1])
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cons_printf ("%s", core->sign->prefix);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
else if (!strcmp ("-", input+1))
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_sign_prefix (core->sign, "");
|
|
|
|
|
else r_sign_prefix (core->sign, input+2);
|
2010-04-08 10:29:47 +00:00
|
|
|
|
break;
|
2010-04-08 13:48:53 +00:00
|
|
|
|
case 'a':
|
2010-04-08 10:29:47 +00:00
|
|
|
|
case 'b':
|
2010-04-08 16:48:39 +00:00
|
|
|
|
case 'f':
|
2010-04-08 13:48:53 +00:00
|
|
|
|
ptr = strchr (input+3, ' ');
|
2010-04-08 10:29:47 +00:00
|
|
|
|
if (ptr) {
|
|
|
|
|
*ptr = 0;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_sign_add (core->sign, core->anal, (int)*input, input+2, ptr+1);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
} else eprintf ("Usage: z%c [name] [bytes]\n", *input);
|
2010-04-08 10:29:47 +00:00
|
|
|
|
break;
|
2010-04-08 13:48:53 +00:00
|
|
|
|
case 'c':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
item = r_sign_check (core->sign, core->block, core->blocksize);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
if (item)
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("f sign.%s @ 0x%08"PFMT64x"\n", item->name, core->offset);
|
2010-04-08 10:29:47 +00:00
|
|
|
|
break;
|
|
|
|
|
case '-':
|
2010-04-08 13:48:53 +00:00
|
|
|
|
if (input[1] == '*') {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_sign_reset (core->sign);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
} else eprintf ("TODO\n");
|
2010-04-08 10:29:47 +00:00
|
|
|
|
break;
|
|
|
|
|
case '/':
|
|
|
|
|
{
|
|
|
|
|
// TODO: parse arg0 and arg1
|
2010-04-08 13:48:53 +00:00
|
|
|
|
ut8 *buf;
|
|
|
|
|
int len, idx;
|
|
|
|
|
ut64 ini, fin;
|
|
|
|
|
RSignItem *si;
|
|
|
|
|
RIOSection *s;
|
|
|
|
|
if (input[1]) {
|
|
|
|
|
char *ptr = strchr (input, ' ');
|
|
|
|
|
if (ptr) {
|
|
|
|
|
*ptr = '\0';
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ini = r_num_math (core->num, input+2);
|
|
|
|
|
fin = r_num_math (core->num, ptr+1);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
} else {
|
|
|
|
|
ini = core->offset;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
fin = ini+r_num_math (core->num, input+2);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
}
|
|
|
|
|
} else {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
s = r_io_section_get (core->io, core->offset);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
if (s) {
|
|
|
|
|
ini = s->vaddr;
|
|
|
|
|
fin = ini + s->size;
|
|
|
|
|
} else {
|
|
|
|
|
eprintf ("No section identified, please provide range.\n");
|
|
|
|
|
return R_FALSE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (ini>=fin) {
|
|
|
|
|
eprintf ("Invalid range.\n");
|
|
|
|
|
return R_FALSE;
|
|
|
|
|
}
|
|
|
|
|
len = fin-ini;
|
|
|
|
|
buf = malloc (len);
|
|
|
|
|
if (buf != NULL) {
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("Ranges are: 0x%08"PFMT64x" 0x%08"PFMT64x"\n", ini, fin);
|
2010-04-08 16:48:39 +00:00
|
|
|
|
r_cons_printf ("f-sign*\n");
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (r_io_read_at (core->io, ini, buf, len) == len) {
|
2010-04-08 13:48:53 +00:00
|
|
|
|
len -= 128;
|
|
|
|
|
for (idx=0; idx<len; idx++) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
si = r_sign_check (core->sign, buf+idx, 128);
|
2010-04-08 16:48:39 +00:00
|
|
|
|
if (si) {
|
|
|
|
|
if (si->type == 'f')
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("f sign.fun_%s_%d @ 0x%08"PFMT64x"\n",
|
2010-04-08 16:48:39 +00:00
|
|
|
|
item->name, idx, core->offset);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
else r_cons_printf ("f sign.%s @ 0x%08"PFMT64x"\n",
|
2010-04-08 13:48:53 +00:00
|
|
|
|
item->name, core->offset);
|
2010-04-08 16:48:39 +00:00
|
|
|
|
}
|
2010-04-08 13:48:53 +00:00
|
|
|
|
}
|
2010-04-14 11:02:23 +00:00
|
|
|
|
} else eprintf ("Cannot read %d bytes at 0x%08"PFMT64x"\n", len, ini);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
free (buf);
|
|
|
|
|
} else eprintf ("Cannot alloc %d bytes\n", len);
|
2010-04-08 10:29:47 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
2010-04-08 13:48:53 +00:00
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_sign_list (core->sign, (input[1]=='*'));
|
2010-04-08 13:48:53 +00:00
|
|
|
|
break;
|
2010-04-08 10:29:47 +00:00
|
|
|
|
default:
|
|
|
|
|
case '?':
|
|
|
|
|
r_cons_printf (
|
2010-04-08 13:48:53 +00:00
|
|
|
|
"Usage: z[abcp/*-] [arg]\n"
|
|
|
|
|
" z show status of zignatures\n"
|
|
|
|
|
" z* display all zignatures\n"
|
|
|
|
|
" zp display current prefix\n"
|
|
|
|
|
" zp prefix define prefix for following zignatures\n"
|
|
|
|
|
" zp- unset prefix\n"
|
2010-04-08 10:29:47 +00:00
|
|
|
|
" z-prefix unload zignatures prefixed as\n"
|
|
|
|
|
" z-* unload all zignatures\n"
|
|
|
|
|
" za ... define new zignature for analysis\n"
|
|
|
|
|
" zb name bytes define new zignature for bytes\n"
|
2010-04-08 16:48:39 +00:00
|
|
|
|
" zf name bytes define new function prelude zignature\n"
|
2010-04-08 14:20:41 +00:00
|
|
|
|
" zg pfx [file] generate siganture for current file\n"
|
2010-04-08 13:48:53 +00:00
|
|
|
|
" .zc @ fcn.foo flag signature if matching (.zc@@fcn)\n"
|
2010-04-08 16:48:39 +00:00
|
|
|
|
" z/ [ini] [end] search zignatures between these regions\n"
|
|
|
|
|
"NOTE: bytes can contain '.' (dots) to specify a binary mask\n");
|
2010-04-08 10:29:47 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static int cmd_iopipe(void *data, const char *input) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
|
|
|
|
switch (input[0]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_lib_list (core->lib);
|
|
|
|
|
r_io_handle_list (core->io);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-01-19 10:25:17 +00:00
|
|
|
|
cmd_io_system (data, input);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 21:12:19 +00:00
|
|
|
|
static void cmd_reg(RCore *core, const char *str) {
|
2010-01-19 10:25:17 +00:00
|
|
|
|
struct r_reg_item_t *r;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
const char *name;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
char *arg;
|
|
|
|
|
int size, i, type = R_REG_TYPE_GPR;
|
|
|
|
|
switch (str[0]) {
|
|
|
|
|
case '?':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (str[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_GPR, R_FALSE);
|
|
|
|
|
r = r_reg_get (core->dbg->reg, str+1, R_REG_TYPE_GPR);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (r == NULL) eprintf ("Unknown register (%s)\n", str+1);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
else r_cons_printf ("0x%08"PFMT64x"\n", r_reg_get_value (core->dbg->reg, r));
|
2010-02-28 21:58:21 +00:00
|
|
|
|
} else
|
2010-01-21 01:38:52 +00:00
|
|
|
|
eprintf ("Usage: dr[*] [type] [size] - get/set registers\n"
|
2010-01-19 10:25:17 +00:00
|
|
|
|
" dr? display this help message\n"
|
2010-03-04 01:11:54 +00:00
|
|
|
|
" dr show registers in rows\n"
|
|
|
|
|
" dr= show registers in columns\n"
|
2010-02-28 21:58:21 +00:00
|
|
|
|
" dr?eax show value of eax register\n"
|
2010-01-19 10:25:17 +00:00
|
|
|
|
" .dr* include common register values in flags\n"
|
|
|
|
|
" .dr- unflag all registers\n"
|
|
|
|
|
" drp [file] load register metadata file\n"
|
|
|
|
|
" drp display current register profile\n"
|
|
|
|
|
" dr show 'gpr' registers\n"
|
|
|
|
|
" drt show all register types\n"
|
2010-02-03 17:15:31 +00:00
|
|
|
|
" drn [pc] get register name for pc,sp,bp,a0-3\n"
|
2010-01-19 10:25:17 +00:00
|
|
|
|
" dr all show all registers\n"
|
|
|
|
|
" dr flg 1 show flag registers ('flg' is type, see drt)\n"
|
|
|
|
|
" dr 16 show 16 bit registers\n"
|
|
|
|
|
" dr 32 show 32 bit registers\n"
|
|
|
|
|
" dr eax=33 set register value. eax = 33\n");
|
2010-01-31 01:30:59 +00:00
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
case 'p':
|
|
|
|
|
if (str[1]) {
|
|
|
|
|
eprintf ("profile: \n");
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (core->dbg->reg_profile)
|
|
|
|
|
r_cons_printf ("%s\n", core->dbg->reg_profile);
|
2010-01-21 01:38:52 +00:00
|
|
|
|
else eprintf ("No register profile defined. Try 'dr.'\n");
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else r_reg_set_profile (core->dbg->reg, str+2);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case 't':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
for (i=0; (name=r_reg_get_type (i));i++)
|
|
|
|
|
r_cons_printf ("%s\n", name);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
2010-02-03 17:15:31 +00:00
|
|
|
|
case 'n':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
name = r_reg_get_name (core->dbg->reg, r_reg_get_name_idx (str+2));
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (name && *name)
|
|
|
|
|
r_cons_printf ("%s\n", name);
|
|
|
|
|
else eprintf ("Oops. try dn [pc|sp|bp|a0|a1|a2|a3]\n");
|
2010-02-03 17:15:31 +00:00
|
|
|
|
break;
|
2010-03-04 01:11:54 +00:00
|
|
|
|
case '=':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_GPR, R_FALSE);
|
|
|
|
|
r_debug_reg_list (core->dbg, R_REG_TYPE_GPR, 32, 2); // XXX detect which one is current usage
|
|
|
|
|
r_debug_reg_list (core->dbg, R_REG_TYPE_GPR, 64, 2);
|
2010-03-04 01:11:54 +00:00
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
case '*':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_GPR, R_FALSE);
|
|
|
|
|
r_debug_reg_list (core->dbg, R_REG_TYPE_GPR, 32, 1); // XXX detect which one is current usage
|
|
|
|
|
r_debug_reg_list (core->dbg, R_REG_TYPE_GPR, 64, 1);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_GPR, R_FALSE);
|
|
|
|
|
r_debug_reg_list (core->dbg, R_REG_TYPE_GPR, 32, 0);
|
|
|
|
|
r_debug_reg_list (core->dbg, R_REG_TYPE_GPR, 64, 0);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case ' ':
|
2010-03-04 01:11:54 +00:00
|
|
|
|
arg = strchr (str+1, '=');
|
2010-01-19 10:25:17 +00:00
|
|
|
|
if (arg) {
|
|
|
|
|
*arg = 0;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r = r_reg_get (core->dbg->reg, str+1, R_REG_TYPE_GPR);
|
2010-02-02 10:09:52 +00:00
|
|
|
|
if (r) {
|
|
|
|
|
//eprintf ("SET(%s)(%s)\n", str, arg+1);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("0x%08"PFMT64x" ->", str,
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_reg_get_value (core->dbg->reg, r));
|
|
|
|
|
r_reg_set_value (core->dbg->reg, r,
|
|
|
|
|
r_num_math (core->num, arg+1));
|
|
|
|
|
r_debug_reg_sync (core->dbg, R_REG_TYPE_GPR, R_TRUE);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("0x%08"PFMT64x"\n",
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_reg_get_value (core->dbg->reg, r));
|
2010-02-02 10:09:52 +00:00
|
|
|
|
} else eprintf ("Unknown register '%s'\n", str+1);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
size = atoi (str+1);
|
|
|
|
|
if (size==0) {
|
|
|
|
|
arg = strchr (str+1, ' ');
|
|
|
|
|
if (arg && size==0) {
|
|
|
|
|
*arg='\0';
|
|
|
|
|
size = atoi (arg);
|
|
|
|
|
} else size = 32;
|
|
|
|
|
eprintf ("ARG(%s)\n", str+1);
|
|
|
|
|
type = r_reg_type_by_name (str+1);
|
|
|
|
|
}
|
|
|
|
|
//printf("type = %d\nsize = %d\n", type, size);
|
|
|
|
|
if (type != R_REG_TYPE_LAST) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_reg_sync (core->dbg, type, R_FALSE);
|
|
|
|
|
r_debug_reg_list (core->dbg, type, size, str[0]=='*');
|
2010-02-05 11:21:37 +00:00
|
|
|
|
} else eprintf ("cmd_reg: Unknown type\n");
|
2010-01-19 10:25:17 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 21:12:19 +00:00
|
|
|
|
static void r_core_cmd_bp(RCore *core, const char *input) {
|
2010-01-19 10:25:17 +00:00
|
|
|
|
if (input[1]==' ')
|
|
|
|
|
input++;
|
|
|
|
|
switch (input[1]) {
|
2010-03-12 17:46:11 +00:00
|
|
|
|
case 't':
|
|
|
|
|
{
|
|
|
|
|
int i = 0;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
RList *list = r_debug_frames (core->dbg);
|
2010-03-12 17:46:11 +00:00
|
|
|
|
RListIter *iter = r_list_iterator (list);
|
|
|
|
|
while (r_list_iter_next (iter)) {
|
|
|
|
|
RDebugFrame *frame = r_list_iter_get (iter);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("%d 0x%08"PFMT64x" %d\n", i++, frame->addr, frame->size);
|
2010-03-12 17:46:11 +00:00
|
|
|
|
}
|
|
|
|
|
r_list_destroy (list);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_list (core->dbg->bp, input[1]=='*');
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case '-':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_del (core->dbg->bp, r_num_math (core->num, input+2));
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'e':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_enable (core->dbg->bp, r_num_math (core->num, input+2), 1);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'd':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_enable (core->dbg->bp, r_num_math (core->num, input+2), 0);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'h':
|
|
|
|
|
if (input[2]==' ') {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (!r_bp_use (core->dbg->bp, input+3))
|
2010-02-01 10:55:56 +00:00
|
|
|
|
eprintf ("Invalid name: '%s'.\n", input+3);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else r_bp_handle_list (core->dbg->bp);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
case '?':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_cons_printf (
|
2010-01-19 10:25:17 +00:00
|
|
|
|
"Usage: db [[-]addr] [len] [rwx] [condstring]\n"
|
|
|
|
|
"db ; list breakpoints\n"
|
|
|
|
|
"db sym.main ; add breakpoint into sym.main\n"
|
|
|
|
|
"db 0x804800 ; add breakpoint\n"
|
|
|
|
|
"db -0x804800 ; remove breakpoint\n"
|
|
|
|
|
"dbe 0x8048000 ; enable breakpoint\n"
|
|
|
|
|
"dbd 0x8048000 ; disable breakpoint\n"
|
2010-03-12 17:46:11 +00:00
|
|
|
|
"dbh x86 ; set/list breakpoint plugin handlers\n"
|
|
|
|
|
"Unrelated:\n"
|
|
|
|
|
"dbt ; debug backtrace\n");
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_add_sw (core->dbg->bp, r_num_math (core->num, input+1),
|
2010-01-19 10:25:17 +00:00
|
|
|
|
1, R_BP_PROT_EXEC);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-04-15 18:24:19 +00:00
|
|
|
|
/* TODO: this should be moved to the core->yank api */
|
2010-03-30 21:12:19 +00:00
|
|
|
|
static int cmd_yank_to(RCore *core, char *arg) {
|
2010-01-12 01:12:18 +00:00
|
|
|
|
ut64 src = core->offset;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
ut64 len = 0;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut64 pos = -1;
|
2009-04-15 18:24:19 +00:00
|
|
|
|
char *str;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut8 *buf;
|
2009-04-15 18:24:19 +00:00
|
|
|
|
|
2010-02-28 21:58:21 +00:00
|
|
|
|
while (*arg==' ')
|
|
|
|
|
arg = arg+1;
|
|
|
|
|
str = strchr (arg, ' ');
|
2009-04-15 18:24:19 +00:00
|
|
|
|
if (str) {
|
|
|
|
|
str[0]='\0';
|
2010-05-19 22:59:42 +00:00
|
|
|
|
len = r_num_math (core->num, arg);
|
|
|
|
|
pos = r_num_math (core->num, str+1);
|
2009-04-15 18:24:19 +00:00
|
|
|
|
str[0]=' ';
|
|
|
|
|
}
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if ((str == NULL) || (pos == -1) || (len == 0)) {
|
2010-03-26 16:01:25 +00:00
|
|
|
|
eprintf ("Usage: yt [len] [dst-addr]\n");
|
2009-04-15 18:24:19 +00:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
#if 0
|
|
|
|
|
if (!config_get("file.write")) {
|
|
|
|
|
eprintf("You are not in read-write mode.\n");
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2010-02-22 23:26:13 +00:00
|
|
|
|
buf = (ut8*)malloc (len);
|
2009-04-15 18:24:19 +00:00
|
|
|
|
r_core_read_at (core, src, buf, len);
|
|
|
|
|
r_core_write_at (core, pos, buf, len);
|
|
|
|
|
free(buf);
|
|
|
|
|
|
2010-01-12 01:12:18 +00:00
|
|
|
|
core->offset = src;
|
2010-02-22 23:26:13 +00:00
|
|
|
|
r_core_block_read (core, 0);
|
2009-04-15 18:24:19 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static int cmd_yank(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
switch (input[0]) {
|
2009-02-18 12:10:59 +00:00
|
|
|
|
case ' ':
|
2010-03-26 16:01:25 +00:00
|
|
|
|
r_core_yank (core, core->offset, atoi(input+1));
|
2009-02-18 12:10:59 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'y':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_core_yank_paste (core, r_num_math(core->num, input+2), 0);
|
2009-02-18 12:10:59 +00:00
|
|
|
|
break;
|
2009-04-15 18:24:19 +00:00
|
|
|
|
case 't':
|
|
|
|
|
{ /* hacky implementation */
|
|
|
|
|
char *arg = strdup(input+1);
|
|
|
|
|
cmd_yank_to(core, arg);
|
|
|
|
|
free(arg);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2009-02-18 12:10:59 +00:00
|
|
|
|
case '\0':
|
2009-09-25 02:04:51 +00:00
|
|
|
|
if (core->yank) {
|
2009-02-18 12:10:59 +00:00
|
|
|
|
int i;
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("0x%08"PFMT64x" %d ",
|
2009-02-18 12:10:59 +00:00
|
|
|
|
core->yank_off, core->yank_len);
|
2010-03-26 16:01:25 +00:00
|
|
|
|
for (i=0;i<core->yank_len;i++)
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf ("%02x", core->yank[i]);
|
|
|
|
|
r_cons_newline ();
|
|
|
|
|
} else eprintf ("No buffer yanked already\n");
|
2009-02-18 12:10:59 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-18 12:10:59 +00:00
|
|
|
|
"Usage: y[y] [len] [[@]addr]\n"
|
|
|
|
|
" y ; show yank buffer information (srcoff len bytes)\n"
|
|
|
|
|
" y 16 ; copy 16 bytes into clipboard\n"
|
|
|
|
|
" y 16 0x200 ; copy 16 bytes into clipboard from 0x200\n"
|
|
|
|
|
" y 16 @ 0x200 ; copy 16 bytes into clipboard from 0x200\n"
|
|
|
|
|
" yy 0x3344 ; paste clipboard\n");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_quit(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-02-01 10:55:56 +00:00
|
|
|
|
switch (input[0]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '?':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: q[!] [retvalue]\n"
|
|
|
|
|
" q ; quit program\n"
|
|
|
|
|
" q! ; force quit (no questions)\n"
|
|
|
|
|
" q 1 ; quit with return value 1\n"
|
|
|
|
|
" q a-b ; quit with return value a-b\n");
|
|
|
|
|
break;
|
|
|
|
|
case ' ':
|
|
|
|
|
case '!':
|
2010-02-28 22:57:55 +00:00
|
|
|
|
input++;
|
|
|
|
|
case '\0':
|
2009-12-30 09:56:41 +00:00
|
|
|
|
// TODO
|
2009-02-05 21:08:46 +00:00
|
|
|
|
default:
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_line_hist_save (".radare2_history");
|
2010-02-28 22:57:55 +00:00
|
|
|
|
if (*input)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_num_math (core->num, input);
|
|
|
|
|
else core->num->value = 0LL;
|
|
|
|
|
//exit (*input?r_num_math (core->num, input+1):0);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
return -2;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
2010-02-28 21:58:21 +00:00
|
|
|
|
return R_FALSE;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static int cmd_interpret(void *data, const char *input) {
|
|
|
|
|
char *str, *ptr, *eol;
|
2010-02-22 23:26:13 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
|
|
|
|
switch (input[0]) {
|
2010-03-19 03:02:23 +00:00
|
|
|
|
case '\0':
|
|
|
|
|
/* repeat last command */
|
2010-03-30 22:03:59 +00:00
|
|
|
|
/* NOTE: Handled in r_core_cmd with ugly strcmp */
|
2010-03-19 03:02:23 +00:00
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case ' ':
|
2010-03-30 22:03:59 +00:00
|
|
|
|
if (!r_core_cmd_file (core, input+1))
|
|
|
|
|
eprintf ("Cannot interpret file.\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '!':
|
|
|
|
|
/* from command */
|
2010-02-22 23:26:13 +00:00
|
|
|
|
r_core_cmd_command (core, input+1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '(':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
//eprintf ("macro call (%s)\n", input+1);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cmd_macro_call (&core->cmd->macro, input+1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '?':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: . [file] | [!command] | [(macro)]\n"
|
|
|
|
|
" . foo.rs ; interpret r script\n"
|
|
|
|
|
" .!rabin -ri $FILE ; interpret output of command\n"
|
2009-02-16 23:09:40 +00:00
|
|
|
|
" .(foo 1 2 3) ; run macro 'foo' with args 1, 2, 3\n"
|
|
|
|
|
" ./m ELF ; interpret output of command /m ELF as r. commands\n");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-02-22 23:26:13 +00:00
|
|
|
|
ptr = str = r_core_cmd_str (core, input);
|
2010-02-01 10:55:56 +00:00
|
|
|
|
for (;;) {
|
2010-02-22 23:26:13 +00:00
|
|
|
|
eol = strchr (ptr, '\n');
|
2009-02-16 23:09:40 +00:00
|
|
|
|
if (eol) eol[0]='\0';
|
2010-02-22 23:26:13 +00:00
|
|
|
|
r_core_cmd (core, ptr, 0);
|
2009-02-16 23:09:40 +00:00
|
|
|
|
if (!eol) break;
|
|
|
|
|
ptr = eol+1;
|
|
|
|
|
}
|
2010-02-22 23:26:13 +00:00
|
|
|
|
free (str);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-01-31 13:22:27 +00:00
|
|
|
|
static int cmd_section(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-02-01 10:55:56 +00:00
|
|
|
|
switch (input[0]) {
|
2010-01-31 13:22:27 +00:00
|
|
|
|
case '?':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
|
|
|
|
" S ; list sections\n"
|
|
|
|
|
" S* ; list sections (in radare commands\n"
|
|
|
|
|
" S= ; list sections (in nice ascii-art bars)\n"
|
2010-04-08 10:29:47 +00:00
|
|
|
|
" S [offset] [vaddr] [size] [vsize] [name] [rwx] ; adds new section\n"
|
2010-02-18 17:58:28 +00:00
|
|
|
|
" S -[offset] ; remove this section definition\n");
|
2010-01-31 13:22:27 +00:00
|
|
|
|
break;
|
|
|
|
|
case ' ':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
switch (input[1]) {
|
2010-01-31 13:22:27 +00:00
|
|
|
|
case '-': // remove
|
|
|
|
|
if (input[2]=='?' || input[2]=='\0')
|
2010-04-08 13:48:53 +00:00
|
|
|
|
eprintf ("Usage: S -N # where N is the section index\n");
|
2010-05-19 22:59:42 +00:00
|
|
|
|
else r_io_section_rm (core->io, atoi (input+1));
|
2010-01-31 13:22:27 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
{
|
2010-04-08 10:29:47 +00:00
|
|
|
|
int i, rwx = 7;
|
2010-01-31 13:22:27 +00:00
|
|
|
|
char *ptr = strdup(input+1);
|
2010-02-18 17:58:28 +00:00
|
|
|
|
const char *name = NULL;
|
|
|
|
|
ut64 vaddr = 0LL;
|
|
|
|
|
ut64 offset = 0LL;
|
|
|
|
|
ut64 size = 0LL;
|
|
|
|
|
ut64 vsize = 0LL;
|
2010-01-31 13:22:27 +00:00
|
|
|
|
|
2010-02-28 21:58:21 +00:00
|
|
|
|
i = r_str_word_set0 (ptr);
|
2010-04-08 10:29:47 +00:00
|
|
|
|
switch (i) {
|
|
|
|
|
case 6: // get rwx
|
|
|
|
|
rwx = r_str_rwx (r_str_word_get0 (ptr, 5));
|
2010-02-18 17:58:28 +00:00
|
|
|
|
case 5: // get name
|
2010-02-28 21:58:21 +00:00
|
|
|
|
name = r_str_word_get0 (ptr, 4);
|
2010-02-18 17:58:28 +00:00
|
|
|
|
case 4: // get vsize
|
2010-05-19 22:59:42 +00:00
|
|
|
|
vsize = r_num_math (core->num, r_str_word_get0 (ptr, 3));
|
2010-02-18 17:58:28 +00:00
|
|
|
|
case 3: // get size
|
2010-05-19 22:59:42 +00:00
|
|
|
|
size = r_num_math (core->num, r_str_word_get0 (ptr, 2));
|
2010-02-18 17:58:28 +00:00
|
|
|
|
case 2: // get vaddr
|
2010-05-19 22:59:42 +00:00
|
|
|
|
vaddr = r_num_math (core->num, r_str_word_get0 (ptr, 1));
|
2010-02-18 17:58:28 +00:00
|
|
|
|
case 1: // get offset
|
2010-05-19 22:59:42 +00:00
|
|
|
|
offset = r_num_math (core->num, r_str_word_get0 (ptr, 0));
|
2010-01-31 13:22:27 +00:00
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_section_add (core->io, offset, vaddr, size, vsize, rwx, name);
|
2010-01-31 13:22:27 +00:00
|
|
|
|
free (ptr);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case '=':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_section_list_visual (core->io, core->offset, core->blocksize);
|
2010-01-31 13:22:27 +00:00
|
|
|
|
break;
|
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_section_list (core->io, core->offset, 0);
|
2010-01-31 13:22:27 +00:00
|
|
|
|
break;
|
|
|
|
|
case '*':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_section_list (core->io, core->offset, 1);
|
2010-01-31 13:22:27 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static int cmd_seek(void *data, const char *input) {
|
2010-02-28 22:57:55 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2009-09-25 02:04:51 +00:00
|
|
|
|
char *cmd, *p;
|
2010-02-28 22:57:55 +00:00
|
|
|
|
ut64 off;
|
|
|
|
|
|
|
|
|
|
if (input[0]=='r') {
|
|
|
|
|
if (input[1] && input[2]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
off = r_debug_reg_get (core->dbg, input+2);
|
2010-02-28 22:57:55 +00:00
|
|
|
|
r_core_seek (core, off, 1);
|
|
|
|
|
} else eprintf ("Usage: 'sr pc' ; seek to register\n");
|
|
|
|
|
} else
|
2010-05-19 00:39:01 +00:00
|
|
|
|
if (input[0]) { // && input[1]) {
|
2010-02-02 10:09:52 +00:00
|
|
|
|
st32 delta = (input[1]==' ')?2:1;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
off = r_num_math (core->num, input + delta);
|
2009-03-12 22:19:58 +00:00
|
|
|
|
if (input[0]==' ' && (input[1]=='+'||input[1]=='-'))
|
|
|
|
|
input = input+1;
|
2010-02-02 10:09:52 +00:00
|
|
|
|
switch (input[0]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case ' ':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_core_seek (core, off, 1);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_sundo_push (core->io);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
break;
|
|
|
|
|
case '*':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_sundo_list (core->io);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '+':
|
2010-05-19 00:39:01 +00:00
|
|
|
|
if (input[1]!='\0') {
|
|
|
|
|
if (input[1]=='+') delta = core->blocksize; else delta = off;
|
|
|
|
|
r_core_seek_delta (core, delta);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else if (r_io_sundo_redo (core->io))
|
|
|
|
|
r_core_seek (core, core->io->off, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '-':
|
2010-05-19 00:39:01 +00:00
|
|
|
|
if (input[1]!='\0') {
|
|
|
|
|
if (input[1]=='-') delta = -core->blocksize; else delta = -off;
|
|
|
|
|
r_core_seek_delta (core, delta);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else if (r_io_sundo (core->io))
|
|
|
|
|
r_core_seek (core, core->io->off, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-03-12 22:19:58 +00:00
|
|
|
|
case 'a':
|
2009-09-25 02:04:51 +00:00
|
|
|
|
off = core->blocksize;
|
|
|
|
|
if (input[1]&&input[2]) {
|
2010-02-01 10:55:56 +00:00
|
|
|
|
cmd = strdup (input);
|
|
|
|
|
p = strchr (cmd+2, ' ');
|
2009-09-25 02:04:51 +00:00
|
|
|
|
if (p) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
off = r_num_math (core->num, p+1);;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
*p = '\0';
|
2009-03-12 22:19:58 +00:00
|
|
|
|
}
|
2010-02-28 21:58:21 +00:00
|
|
|
|
cmd[0] = 's';
|
2009-09-25 02:04:51 +00:00
|
|
|
|
// perform real seek if provided
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cmd_call (core->cmd, cmd);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
free (cmd);
|
2009-03-12 22:19:58 +00:00
|
|
|
|
}
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_core_seek_align (core, off, 0);
|
2009-03-12 22:19:58 +00:00
|
|
|
|
break;
|
2010-03-03 17:05:24 +00:00
|
|
|
|
case 'b':
|
|
|
|
|
r_core_anal_bb_seek (core, off);
|
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '?':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: s[+-] [addr]\n"
|
2009-03-12 22:19:58 +00:00
|
|
|
|
" s 0x320 ; seek to this address\n"
|
2010-05-19 00:39:01 +00:00
|
|
|
|
" s- ; undo seek\n"
|
|
|
|
|
" s+ ; redo seek\n"
|
|
|
|
|
" s* ; list undo seek history\n"
|
2009-03-12 22:19:58 +00:00
|
|
|
|
" s++ ; seek blocksize bytes forward\n"
|
|
|
|
|
" s-- ; seek blocksize bytes backward\n"
|
|
|
|
|
" s+ 512 ; seek 512 bytes forward\n"
|
|
|
|
|
" s- 512 ; seek 512 bytes backward\n"
|
2010-03-03 17:05:24 +00:00
|
|
|
|
" sa [[+-]a] [asz] ; seek asz (or bsize) aligned to addr\n"
|
|
|
|
|
" sb ; seek aligned to bb start\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2010-04-14 11:02:23 +00:00
|
|
|
|
} else r_cons_printf ("0x%"PFMT64x"\n", core->offset);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_help(void *data, const char *input) {
|
|
|
|
|
RCore *core = (RCore *)data;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut64 n;
|
2010-02-02 10:09:52 +00:00
|
|
|
|
switch (input[0]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case ' ':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
n = r_num_math (core->num, input+1);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("%"PFMT64d" 0x%"PFMT64x"\n", n,n);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '=':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_num_math (core->num, input+1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '+':
|
|
|
|
|
if (input[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (core->num->value & UT64_GT0)
|
2010-01-31 13:22:27 +00:00
|
|
|
|
r_core_cmd (core, input+1, 0);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else r_cons_printf ("0x%"PFMT64x"\n", core->num->value);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '-':
|
|
|
|
|
if (input[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (core->num->value & UT64_LT0)
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_core_cmd (core, input+1, 0);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else r_cons_printf ("0x%"PFMT64x"\n", core->num->value);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '!': // ??
|
|
|
|
|
if (input[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (core->num->value != UT64_MIN)
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_core_cmd (core, input+1, 0);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else r_cons_printf ("0x%"PFMT64x"\n", core->num->value);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-04-15 00:29:01 +00:00
|
|
|
|
case '$':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
return cmd_help (data, " $?");
|
2009-03-13 11:28:36 +00:00
|
|
|
|
case 'z':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
for (input=input+1;input[0]==' ';input=input+1);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->num->value = strlen(input);
|
2009-03-13 11:28:36 +00:00
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
case 't': {
|
|
|
|
|
struct r_prof_t prof;
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_prof_start (&prof);
|
|
|
|
|
r_core_cmd (core, input+1, 0);
|
|
|
|
|
r_prof_end (&prof);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->num->value = (ut64)prof.result;
|
2010-02-02 10:09:52 +00:00
|
|
|
|
eprintf ("%lf\n", prof.result);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
} break;
|
2009-03-13 11:28:36 +00:00
|
|
|
|
case '?': // ???
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (input[1]=='?') {
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: ?[?[?]] expression\n"
|
|
|
|
|
" ? eip-0x804800 ; calculate result for this math expr\n"
|
2009-03-13 11:28:36 +00:00
|
|
|
|
" ?= eip-0x804800 ; same as above without user feedback\n"
|
|
|
|
|
" ?? [cmd] ; ? == 0 run command when math matches\n"
|
|
|
|
|
" ?z str ; returns the length of string (0 if null)\n"
|
|
|
|
|
" ?t cmd ; returns the time to run a command\n"
|
|
|
|
|
" ?! [cmd] ; ? != 0\n"
|
|
|
|
|
" ?+ [cmd] ; ? > 0\n"
|
|
|
|
|
" ?- [cmd] ; ? < 0\n"
|
2009-04-15 18:24:19 +00:00
|
|
|
|
" ??? ; show this help\n"
|
|
|
|
|
"$variables:\n"
|
|
|
|
|
" $$ = here (current seek)\n"
|
|
|
|
|
" $s = file size\n"
|
|
|
|
|
" $b = block size\n"
|
|
|
|
|
" $j = jump address\n"
|
|
|
|
|
" $f = address of next opcode\n"
|
|
|
|
|
" $r = opcode reference pointer\n"
|
|
|
|
|
" $e = 1 if end of block, else 0\n"
|
|
|
|
|
" ${eval} = get value of eval variable\n"
|
|
|
|
|
" $? = last comparision value\n");
|
|
|
|
|
return 0;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
} else
|
|
|
|
|
if (input[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (core->num->value == UT64_MIN)
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_core_cmd (core, input+1, 0);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else r_cons_printf ("0x%"PFMT64x"\n", core->num->value);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '\0':
|
|
|
|
|
default:
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage:\n"
|
2009-02-18 12:10:59 +00:00
|
|
|
|
" a ; perform analysis of code\n"
|
|
|
|
|
" b [bsz] ; get or change block size\n"
|
|
|
|
|
" C[CFf..] ; Code metadata management\n"
|
2009-03-14 11:39:37 +00:00
|
|
|
|
" d[hrscb] ; debugger commands\n"
|
2009-02-18 12:10:59 +00:00
|
|
|
|
" e [a[=b]] ; list/get/set config evaluable vars\n"
|
|
|
|
|
" f [name][sz][at] ; set flag at current address\n"
|
|
|
|
|
" s [addr] ; seek to address\n"
|
2010-01-31 13:22:27 +00:00
|
|
|
|
" S?[size] [vaddr] ; IO section manipulation information\n"
|
2009-02-18 12:10:59 +00:00
|
|
|
|
" i [file] ; get info about opened file\n"
|
2010-02-01 10:55:56 +00:00
|
|
|
|
" o [file] (addr) ; open file at optional address\n"
|
2009-02-18 12:10:59 +00:00
|
|
|
|
" p?[len] ; print current block with format and length\n"
|
2009-03-14 11:39:37 +00:00
|
|
|
|
" V[vcmds] ; enter visual mode (vcmds=visualvisual keystrokes)\n"
|
|
|
|
|
" w[mode] [arg] ; multiple write operations\n"
|
2010-05-03 19:24:58 +00:00
|
|
|
|
" x [len] ; alias for 'px' (print hexadecimal)\n"
|
2009-02-18 12:10:59 +00:00
|
|
|
|
" y [len] [off] ; yank/paste bytes from/to memory\n"
|
|
|
|
|
" ? [expr] ; help or evaluate math expression\n"
|
|
|
|
|
" /[xmp/] ; search for bytes, regexps, patterns, ..\n"
|
|
|
|
|
" |[cmd] ; run this command thru the io pipe (no args=list)\n"
|
|
|
|
|
" #[algo] [len] ; calculate hash checksum of current block\n"
|
|
|
|
|
" .[ file|!cmd|cmd|(macro)] ; interpret as radare cmds\n"
|
|
|
|
|
" (macro arg0 arg1) ; define scripting macros\n"
|
|
|
|
|
" q [ret] ; quit program with a return value\n"
|
2009-04-15 00:29:01 +00:00
|
|
|
|
"Use '?$' to get help for the variables\n"
|
2009-09-24 10:29:05 +00:00
|
|
|
|
"Use '?""?""?' for extra help about '?' subcommands.\n"
|
2009-02-18 12:10:59 +00:00
|
|
|
|
"Append '?' to any char command to get detailed help\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_bsize(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-02-02 10:09:52 +00:00
|
|
|
|
switch (input[0]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '\0':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_cons_printf ("0x%x\n", core->blocksize);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
//input = r_str_clean(input);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_core_block_size (core, r_num_math (core->num, input));
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2010-05-19 00:39:01 +00:00
|
|
|
|
|
2010-03-26 10:59:28 +00:00
|
|
|
|
// move it out // r_diff maybe?
|
|
|
|
|
static int radare_compare(RCore *core, const ut8 *f, const ut8 *d, int len) {
|
|
|
|
|
int i, eq = 0;
|
|
|
|
|
for (i=0;i<len;i++) {
|
|
|
|
|
if (f[i]==d[i]) {
|
|
|
|
|
eq++;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("0x%08"PFMT64x" (byte=%.2d) %02x '%c' -> %02x '%c'\n",
|
2010-03-26 10:59:28 +00:00
|
|
|
|
core->offset+i, i+1,
|
|
|
|
|
f[i], (IS_PRINTABLE(f[i]))?f[i]:' ',
|
|
|
|
|
d[i], (IS_PRINTABLE(d[i]))?d[i]:' ');
|
|
|
|
|
}
|
|
|
|
|
eprintf ("Compare %d/%d equal bytes\n", eq, len);
|
|
|
|
|
return len-eq;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-05 11:21:37 +00:00
|
|
|
|
static int cmd_cmp(void *data, const char *input) {
|
|
|
|
|
RCore *core = data;
|
2010-03-26 10:59:28 +00:00
|
|
|
|
FILE *fd;
|
|
|
|
|
ut8 *buf;
|
|
|
|
|
int ret;
|
|
|
|
|
ut32 v32;
|
|
|
|
|
ut64 v64;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
|
|
|
|
|
switch (*input) {
|
2010-03-26 10:59:28 +00:00
|
|
|
|
case ' ':
|
|
|
|
|
radare_compare (core, core->block, (ut8*)input+1, strlen (input+1)+1);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
2010-03-26 10:59:28 +00:00
|
|
|
|
case 'x':
|
|
|
|
|
if (input[1]!=' ') {
|
|
|
|
|
eprintf ("Usage: cx 001122'\n");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
buf = (ut8*)malloc (strlen (input+2));
|
|
|
|
|
ret = r_hex_str2bin (input+2, buf);
|
|
|
|
|
if (ret<1) {
|
|
|
|
|
eprintf ("Cannot parse hexpair\n");
|
|
|
|
|
} else radare_compare (core, core->block, buf, ret);
|
|
|
|
|
free (buf);
|
|
|
|
|
break;
|
|
|
|
|
case 'X':
|
|
|
|
|
buf = malloc (core->blocksize);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ret = r_io_read_at (core->io, r_num_math (core->num, input+1), buf, core->blocksize);
|
2010-03-26 10:59:28 +00:00
|
|
|
|
radare_compare (core, core->block, buf, ret);
|
|
|
|
|
free(buf);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'f':
|
|
|
|
|
if (input[1]!=' ') {
|
2010-03-26 10:59:28 +00:00
|
|
|
|
eprintf ("Please. use 'cf [file]'\n");
|
2010-02-05 11:21:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2010-03-26 10:59:28 +00:00
|
|
|
|
fd = fopen (input+2, "rb");
|
2010-02-05 11:21:37 +00:00
|
|
|
|
if (fd == NULL) {
|
2010-03-26 10:59:28 +00:00
|
|
|
|
eprintf ("Cannot open file '%s'\n", input+2);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2010-03-26 10:59:28 +00:00
|
|
|
|
buf = (ut8 *)malloc (core->blocksize);
|
|
|
|
|
fread (buf, 1, core->blocksize, fd);
|
|
|
|
|
fclose (fd);
|
|
|
|
|
radare_compare (core, core->block, buf, core->blocksize);
|
|
|
|
|
free (buf);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
2010-03-26 10:59:28 +00:00
|
|
|
|
case 'q':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
v64 = (ut64) r_num_math (core->num, input+1);
|
2010-03-26 16:01:25 +00:00
|
|
|
|
radare_compare (core, core->block, (ut8*)&v64, sizeof (v64));
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
2010-03-26 10:59:28 +00:00
|
|
|
|
case 'd':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
v32 = (ut32) r_num_math (core->num, input+1);
|
2010-03-26 16:01:25 +00:00
|
|
|
|
radare_compare (core, core->block, (ut8*)&v32, sizeof (v32));
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
2010-03-26 10:59:28 +00:00
|
|
|
|
#if 0
|
|
|
|
|
case 'c':
|
|
|
|
|
radare_compare_code (
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_num_math (core->num, input+1),
|
2010-03-26 10:59:28 +00:00
|
|
|
|
core->block, core->blocksize);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'D':
|
2010-03-26 10:59:28 +00:00
|
|
|
|
{ // XXX ugly hack
|
2010-02-05 11:21:37 +00:00
|
|
|
|
char cmd[1024];
|
2010-03-26 10:59:28 +00:00
|
|
|
|
sprintf (cmd, "radiff -b %s %s", ".curblock", input+2);
|
|
|
|
|
r_file_dump (".curblock", config.block, config.block_size);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
radare_system(cmd);
|
|
|
|
|
unlink(".curblock");
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
#endif
|
|
|
|
|
case '?':
|
|
|
|
|
r_cons_strcat (
|
|
|
|
|
"Usage: c[?cdfx] [argument]\n"
|
|
|
|
|
" c [string] Compares a plain with escaped chars string\n"
|
2010-03-26 10:59:28 +00:00
|
|
|
|
//" cc [offset] Code bindiff current block against offset\n"
|
2010-02-05 11:21:37 +00:00
|
|
|
|
" cd [value] Compare a doubleword from a math expression\n"
|
2010-03-26 10:59:28 +00:00
|
|
|
|
//" cD [file] Like above, but using radiff -b\n");
|
|
|
|
|
" cq [value] Compare a quadword from a math expression\n"
|
2010-02-05 11:21:37 +00:00
|
|
|
|
" cx [hexpair] Compare hexpair string\n"
|
|
|
|
|
" cX [addr] Like 'cc' but using hexdiff output\n"
|
2010-03-26 16:01:25 +00:00
|
|
|
|
" cf [file] Compare contents of file at current seek\n");
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-03-19 11:00:04 +00:00
|
|
|
|
eprintf ("Usage: c[?Ddxf] [argument]\n");
|
2010-02-05 11:21:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int cmd_info(void *data, const char *input) {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2009-04-01 10:41:42 +00:00
|
|
|
|
char buf[1024];
|
2010-02-05 11:21:37 +00:00
|
|
|
|
switch (*input) {
|
2009-04-01 10:41:42 +00:00
|
|
|
|
case 's':
|
|
|
|
|
case 'i':
|
|
|
|
|
case 'I':
|
|
|
|
|
case 'e':
|
|
|
|
|
case 'S':
|
|
|
|
|
case 'z':
|
2010-02-21 10:35:49 +00:00
|
|
|
|
snprintf (buf, sizeof (buf), "rabin2 -%c%s%s '%s'", input[0],
|
2010-05-19 22:59:42 +00:00
|
|
|
|
input[1]=='*'?"r":"", core->io->va?"v":"", core->file->filename);
|
2010-02-01 10:55:56 +00:00
|
|
|
|
eprintf ("(%s)\n", buf);
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_sys_cmd (buf);
|
2009-04-01 10:41:42 +00:00
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '?':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_cons_printf (
|
2009-04-01 10:41:42 +00:00
|
|
|
|
"Usage: i[eiIsSz]* ; get info from opened file (rabin2)\n"
|
|
|
|
|
"; Append a '*' to get the output in radare commands\n"
|
|
|
|
|
" ii ; imports\n"
|
|
|
|
|
" iI ; binary info\n"
|
|
|
|
|
" ie ; entrypoint\n"
|
|
|
|
|
" is ; symbols\n"
|
|
|
|
|
" iS ; sections\n"
|
|
|
|
|
" iz ; strings\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '*':
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf ("uri: %s\n", core->file->uri);
|
|
|
|
|
r_cons_printf ("filesize: 0x%x\n", core->file->size);
|
|
|
|
|
r_cons_printf ("blocksize: 0x%x\n", core->blocksize);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_print(void *data, const char *input) {
|
|
|
|
|
RCore *core = (RCore *)data;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
int l, len = core->blocksize;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut32 tbs = core->blocksize;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
/* XXX: This is only for pd/pD ??? */
|
2010-05-07 11:00:52 +00:00
|
|
|
|
/* XXX: pd must change bs too */
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (input[0] && input[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
l = (int) r_num_math (core->num, input+2);
|
2010-03-01 09:49:04 +00:00
|
|
|
|
// exceptions are: disasm and memoryfmt */
|
|
|
|
|
if (input[0] != 'd' && input[0] != 'm') {
|
2010-02-12 02:17:32 +00:00
|
|
|
|
if (l>0) len = l;
|
|
|
|
|
if (l>tbs) r_core_block_size (core, l);
|
2010-04-14 20:28:45 +00:00
|
|
|
|
l = len;
|
2010-02-12 02:17:32 +00:00
|
|
|
|
}
|
2010-04-14 20:28:45 +00:00
|
|
|
|
} else l = len;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
2010-03-19 00:45:52 +00:00
|
|
|
|
switch (input[0]) {
|
2010-02-12 02:17:32 +00:00
|
|
|
|
case 'D':
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 'd':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_disasm (core->print, core, core->offset, core->block, len, l);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-02-09 00:54:09 +00:00
|
|
|
|
case 's':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_string (core->print, core->offset, core->block, len, 0, 1, 0); //, 78, 1);
|
2009-03-12 01:42:35 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'S':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_string (core->print, core->offset, core->block, len, 1, 1, 0); //, 78, 1);
|
2009-03-12 01:42:35 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'u':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_string (core->print, core->offset, core->block, len, 0, 1, 1); //, 78, 1);
|
2009-03-12 01:42:35 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'U':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_string (core->print, core->offset, core->block, len, 1, 1, 1); //, 78, 1);
|
2009-02-09 00:54:09 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'c':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_code (core->print, core->offset, core->block, len); //, 78, 1);
|
2009-02-09 00:54:09 +00:00
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 'r':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_raw (core->print, core->block, len);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-03-12 01:42:35 +00:00
|
|
|
|
case 'o':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_hexdump (core->print, core->offset, core->block, len, 8, 1); //, 78, !(input[1]=='-'));
|
2009-03-12 01:42:35 +00:00
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 'x':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_hexdump (core->print, core->offset, core->block, len, 16, 1); //, 78, !(input[1]=='-'));
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '8':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_bytes (core->print, core->block, len, "%02x");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2010-03-01 09:49:04 +00:00
|
|
|
|
case 'm':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_print_format (core->print, core->offset, core->block, len, input+1);
|
2010-03-01 09:49:04 +00:00
|
|
|
|
break;
|
2010-03-18 23:51:16 +00:00
|
|
|
|
case 'n': // easter penis
|
|
|
|
|
for (l=0; l<10; l++) {
|
|
|
|
|
printf ("\r8");
|
|
|
|
|
for (len=0;len<l;len++)
|
|
|
|
|
printf ("=");
|
|
|
|
|
printf ("D");
|
|
|
|
|
r_sys_usleep (100000);
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
}
|
|
|
|
|
for (l=0; l<3; l++) {
|
|
|
|
|
printf ("~");
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
r_sys_usleep (100000);
|
|
|
|
|
}
|
|
|
|
|
printf ("\n");
|
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
default:
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Usage: p[fmt] [len]\n"
|
2009-02-05 21:08:46 +00:00
|
|
|
|
" p8 [len] 8bit hexpair list of bytes\n"
|
|
|
|
|
" px [len] hexdump of N bytes\n"
|
2009-03-12 01:42:35 +00:00
|
|
|
|
" po [len] octal dump of N bytes\n"
|
2009-02-09 00:54:09 +00:00
|
|
|
|
" pc [len] output C format\n"
|
|
|
|
|
" ps [len] print string\n"
|
2010-03-01 09:49:04 +00:00
|
|
|
|
" pm [fmt] print formatted memory\n" // TODO: rename to pf??
|
2009-03-12 01:42:35 +00:00
|
|
|
|
" pS [len] print wide string\n"
|
2010-02-12 02:17:32 +00:00
|
|
|
|
" pd [len] disassemble N opcodes\n"
|
|
|
|
|
" pD [len] disassemble N bytes\n"
|
2009-03-12 01:42:35 +00:00
|
|
|
|
" pr [len] print N raw bytes\n"
|
|
|
|
|
" pu [len] print N url encoded bytes\n"
|
2010-02-28 21:58:21 +00:00
|
|
|
|
" pU [len] print N wide url encoded bytes\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (tbs != core->blocksize)
|
2010-02-03 17:15:31 +00:00
|
|
|
|
r_core_block_size (core, tbs);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static int cmd_hexdump(void *data, const char *input) {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
return cmd_print (data, input-1);
|
2009-04-15 18:24:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static int cmd_flag(void *data, const char *input) {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
|
|
|
|
int len = strlen (input)+1;
|
|
|
|
|
char *str = alloca (len);
|
|
|
|
|
memcpy (str, input+1, len);
|
2009-03-09 13:08:53 +00:00
|
|
|
|
|
2010-02-28 21:58:21 +00:00
|
|
|
|
switch (input[0]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '+':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_flag_set (core->flags, str, core->offset, core->blocksize, 1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-03-09 13:08:53 +00:00
|
|
|
|
case ' ': {
|
|
|
|
|
char *s = NULL, *s2 = NULL;
|
2010-01-12 01:12:18 +00:00
|
|
|
|
ut64 seek = core->offset;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut32 bsze = core->blocksize;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
s = strchr (str, ' ');
|
2009-03-09 13:08:53 +00:00
|
|
|
|
if (s) {
|
|
|
|
|
*s = '\0';
|
2010-03-30 23:06:26 +00:00
|
|
|
|
s2 = strchr (s+1, ' ');
|
2009-03-09 13:08:53 +00:00
|
|
|
|
if (s2) {
|
|
|
|
|
*s2 = '\0';
|
2010-03-30 23:06:26 +00:00
|
|
|
|
if (s2[1]&&s2[2])
|
2010-05-19 22:59:42 +00:00
|
|
|
|
seek = r_num_math (core->num, s2+1);
|
2009-03-09 13:08:53 +00:00
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
bsze = r_num_math (core->num, s+1);
|
2009-03-09 13:08:53 +00:00
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_flag_set (core->flags, str, seek, bsze, 0);
|
2009-03-09 13:08:53 +00:00
|
|
|
|
if (s) *s=' ';
|
|
|
|
|
if (s2) *s2=' ';
|
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '-':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_flag_unset (core->flags, input+1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2010-04-09 09:32:17 +00:00
|
|
|
|
case 'S':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_flag_sort (core->flags, (input[1]=='n'));
|
2010-04-09 09:32:17 +00:00
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 's':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (input[1]==' ') r_flag_space_set (core->flags, input+2);
|
|
|
|
|
else r_flag_space_list (core->flags);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-03-27 00:16:27 +00:00
|
|
|
|
case 'o':
|
|
|
|
|
{
|
2009-04-02 13:07:26 +00:00
|
|
|
|
char *file = PREFIX"/share/doc/radare2/fortunes";
|
2009-03-27 00:16:27 +00:00
|
|
|
|
char *line = r_file_slurp_random_line (file);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_cons_printf (" -- %s\n", line);
|
|
|
|
|
free (line);
|
2009-03-27 00:16:27 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '*':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_flag_list (core->flags, 1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_flag_list (core->flags, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '?':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
2010-02-28 21:58:21 +00:00
|
|
|
|
"Usage: f[?] [flagname]\n"
|
2009-02-05 21:08:46 +00:00
|
|
|
|
" f name 12 @ 33 ; set flag 'name' with size 12 at 33\n"
|
|
|
|
|
" f name 12 33 ; same as above\n"
|
|
|
|
|
" f+name 12 @ 33 ; like above but creates new one if doesnt exist\n"
|
|
|
|
|
" f-name ; remove flag 'name'\n"
|
|
|
|
|
" f ; list flags\n"
|
2010-04-09 09:32:17 +00:00
|
|
|
|
" f* ; list flags in r commands\n"
|
|
|
|
|
" fs functions ; set flagspace\n"
|
|
|
|
|
" fs * ; set no flagspace\n"
|
|
|
|
|
" fs ; display flagspaces\n"
|
|
|
|
|
" fS[on] ; sort flags by offset or name\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-04 00:46:25 +00:00
|
|
|
|
static void cmd_syscall_do(RCore *core, int num) {
|
|
|
|
|
int i;
|
2010-05-19 00:39:01 +00:00
|
|
|
|
char str[64];
|
2010-05-19 22:59:42 +00:00
|
|
|
|
RSyscallItem *item = r_syscall_get (core->syscall, num, -1);
|
2010-03-04 00:46:25 +00:00
|
|
|
|
r_cons_printf ("%d = %s (", item->num, item->name);
|
|
|
|
|
// TODO: move this to r_syscall
|
|
|
|
|
for (i=0; i<item->args; i++) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ut64 arg = r_debug_arg_get (core->dbg, R_TRUE, i+1);
|
2010-03-04 00:46:25 +00:00
|
|
|
|
if (item->sargs==NULL)
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("0x%08"PFMT64x"", arg);
|
2010-03-04 00:46:25 +00:00
|
|
|
|
else
|
|
|
|
|
switch (item->sargs[i]) {
|
|
|
|
|
case 'p': // pointer
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("0x%08"PFMT64x"", arg);
|
2010-03-04 00:46:25 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'i':
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("%"PFMT64d"", arg);
|
2010-03-04 00:46:25 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'z':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_read_at (core->io, arg, (ut8*)str, sizeof (str));
|
2010-03-04 00:46:25 +00:00
|
|
|
|
// TODO: filter zero terminated string
|
|
|
|
|
str[63] = '\0';
|
|
|
|
|
r_str_filter (str, strlen (str));
|
|
|
|
|
r_cons_printf ("\"%s\"", str);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("0x%08"PFMT64x"", arg);
|
2010-03-04 00:46:25 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (i+1<item->args)
|
|
|
|
|
r_cons_printf (", ");
|
|
|
|
|
}
|
|
|
|
|
r_cons_printf (")\n");
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_anal(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2009-02-15 23:57:03 +00:00
|
|
|
|
int l, len = core->blocksize;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut32 tbs = core->blocksize;
|
2009-02-15 23:57:03 +00:00
|
|
|
|
|
|
|
|
|
if (input[0] && input[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
l = (int) r_num_get (core->num, input+2);
|
2009-02-15 23:57:03 +00:00
|
|
|
|
if (l>0) len = l;
|
|
|
|
|
if (l>tbs) {
|
2010-03-04 00:46:25 +00:00
|
|
|
|
r_core_block_size (core, l);
|
2009-02-15 23:57:03 +00:00
|
|
|
|
len = l;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-24 00:24:27 +00:00
|
|
|
|
switch (input[0]) {
|
2010-01-31 01:30:59 +00:00
|
|
|
|
case 'h':
|
2010-02-26 12:08:42 +00:00
|
|
|
|
if (input[1]) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (!r_anal_use (core->anal, input+2))
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("Cannot use '%s' anal plugin.\n", input+2);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else r_anal_list (core->anal);
|
2009-02-15 23:57:03 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'o':
|
2009-02-16 01:12:02 +00:00
|
|
|
|
{
|
|
|
|
|
int ret, idx;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut8 *buf = core->block;
|
2009-03-31 22:32:26 +00:00
|
|
|
|
struct r_anal_aop_t aop;
|
2009-02-16 01:12:02 +00:00
|
|
|
|
|
2010-03-26 16:01:25 +00:00
|
|
|
|
for (idx=ret=0; idx<len; idx+=ret) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ret = r_anal_aop (core->anal, &aop,
|
2010-02-22 03:02:13 +00:00
|
|
|
|
core->offset+idx, buf + idx, (len-idx));
|
2010-03-26 16:01:25 +00:00
|
|
|
|
if (ret<1) {
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("Oops at 0x%08"PFMT64x"\n", core->offset+idx);
|
2010-03-26 16:01:25 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("addr: 0x%08"PFMT64x"\n", core->offset+idx);
|
2010-03-26 16:01:25 +00:00
|
|
|
|
r_cons_printf ("size: %d\n", aop.length);
|
|
|
|
|
r_cons_printf ("type: %d\n", aop.type); // TODO: string
|
|
|
|
|
r_cons_printf ("eob: %d\n", aop.eob);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("jump: 0x%08"PFMT64x"\n", aop.jump);
|
|
|
|
|
r_cons_printf ("fail: 0x%08"PFMT64x"\n", aop.fail);
|
2010-03-26 16:01:25 +00:00
|
|
|
|
r_cons_printf ("stack: %d\n", aop.stackop); // TODO: string
|
|
|
|
|
r_cons_printf ("cond: %d\n", aop.cond); // TODO: string
|
|
|
|
|
r_cons_printf ("family: %d\n", aop.family);
|
|
|
|
|
r_cons_printf ("\n");
|
2010-04-14 11:02:23 +00:00
|
|
|
|
//r_cons_printf ("false: 0x%08"PFMT64x"\n", core->offset+idx);
|
2010-03-26 16:01:25 +00:00
|
|
|
|
}
|
2009-02-16 01:12:02 +00:00
|
|
|
|
}
|
2009-02-15 23:57:03 +00:00
|
|
|
|
break;
|
2010-03-01 19:13:36 +00:00
|
|
|
|
case 'b':
|
2010-03-03 02:03:01 +00:00
|
|
|
|
switch (input[1]) {
|
|
|
|
|
case '-':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_anal_bb_del (core->anal, r_num_math (core->num, input+2));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
break;
|
|
|
|
|
case '+':
|
|
|
|
|
{
|
|
|
|
|
char *ptr = strdup(input+3);
|
|
|
|
|
ut64 addr = -1LL;
|
|
|
|
|
ut64 size = 0LL;
|
|
|
|
|
ut64 jump = -1LL;
|
|
|
|
|
ut64 fail = -1LL;
|
|
|
|
|
|
|
|
|
|
switch(r_str_word_set0 (ptr)) {
|
|
|
|
|
case 4: // get fail
|
2010-05-19 22:59:42 +00:00
|
|
|
|
fail = r_num_math (core->num, r_str_word_get0 (ptr, 3));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
case 3: // get jump
|
2010-05-19 22:59:42 +00:00
|
|
|
|
jump = r_num_math (core->num, r_str_word_get0 (ptr, 2));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
case 2: // get size
|
2010-05-19 22:59:42 +00:00
|
|
|
|
size = r_num_math (core->num, r_str_word_get0 (ptr, 1));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
case 1: // get addr
|
2010-05-19 22:59:42 +00:00
|
|
|
|
addr = r_num_math (core->num, r_str_word_get0 (ptr, 0));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (!r_anal_bb_add (core->anal, addr, size, jump, fail))
|
2010-03-03 02:03:01 +00:00
|
|
|
|
eprintf ("Cannot add bb (duplicated or overlaped)\n");
|
|
|
|
|
free (ptr);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'l':
|
|
|
|
|
r_core_anal_bb_list (core, 0);
|
|
|
|
|
break;
|
|
|
|
|
case '*':
|
|
|
|
|
r_core_anal_bb_list (core, 1);
|
|
|
|
|
break;
|
|
|
|
|
case '?':
|
|
|
|
|
r_cons_printf (
|
2010-03-03 02:12:24 +00:00
|
|
|
|
"Usage: ab[?+-l*]\n"
|
2010-03-03 02:03:01 +00:00
|
|
|
|
" ab @ [addr] ; Analyze basic blocks (start at addr)\n"
|
2010-03-03 17:05:24 +00:00
|
|
|
|
" ab+ addr size [jump] [fail] ; Add basic block\n"
|
2010-03-03 02:03:01 +00:00
|
|
|
|
" ab- [addr] ; Clean all basic block data (or bb at addr and childs)\n"
|
|
|
|
|
" abl ; List basic blocks\n"
|
|
|
|
|
" ab* ; Output radare commands\n");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
r_core_anal_bb (core, core->offset,
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_get_i (core->config, "anal.depth"));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'f':
|
|
|
|
|
switch (input[1]) {
|
|
|
|
|
case '-':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_anal_fcn_del (core->anal, r_num_math (core->num, input+2));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
break;
|
|
|
|
|
case '+':
|
2010-03-03 11:08:27 +00:00
|
|
|
|
{
|
|
|
|
|
char *ptr = strdup(input+3);
|
|
|
|
|
const char *name = NULL;
|
|
|
|
|
ut64 addr = -1LL;
|
|
|
|
|
ut64 size = 0LL;
|
|
|
|
|
|
2010-03-03 17:05:24 +00:00
|
|
|
|
if (r_str_word_set0 (ptr) == 3) {
|
2010-03-03 11:08:27 +00:00
|
|
|
|
name = r_str_word_get0 (ptr, 2);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
size = r_num_math (core->num, r_str_word_get0 (ptr, 1));
|
|
|
|
|
addr = r_num_math (core->num, r_str_word_get0 (ptr, 0));
|
|
|
|
|
if (!r_anal_fcn_add (core->anal, addr, size, name))
|
2010-03-03 17:05:24 +00:00
|
|
|
|
eprintf ("Cannot add function (duplicated or overlaped)\n");
|
2010-03-03 11:08:27 +00:00
|
|
|
|
}
|
|
|
|
|
free (ptr);
|
|
|
|
|
}
|
2010-03-03 02:12:24 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'l':
|
2010-03-03 11:08:27 +00:00
|
|
|
|
r_core_anal_fcn_list (core, 0);
|
2010-03-03 02:12:24 +00:00
|
|
|
|
break;
|
|
|
|
|
case '*':
|
2010-03-03 11:08:27 +00:00
|
|
|
|
r_core_anal_fcn_list (core, 1);
|
2010-03-03 02:12:24 +00:00
|
|
|
|
break;
|
|
|
|
|
case '?':
|
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Usage: af[?+-l*]\n"
|
|
|
|
|
" af @ [addr] ; Analyze functions (start at addr)\n"
|
2010-03-03 17:05:24 +00:00
|
|
|
|
" af+ addr size name ; Add function\n"
|
2010-03-03 02:12:24 +00:00
|
|
|
|
" af- [addr] ; Clean all function analysis data (or function at addr)\n"
|
|
|
|
|
" afl ; List functions\n"
|
|
|
|
|
" af* ; Output radare commands\n");
|
2010-03-03 02:03:01 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-03-03 11:08:27 +00:00
|
|
|
|
r_core_anal_fcn (core, core->offset, -1,
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_get_i (core->config, "anal.depth"));
|
2010-03-03 02:03:01 +00:00
|
|
|
|
}
|
2010-02-26 12:08:42 +00:00
|
|
|
|
break;
|
2010-02-27 10:56:41 +00:00
|
|
|
|
case 'g':
|
2010-03-03 13:35:18 +00:00
|
|
|
|
switch (input[1]) {
|
2010-03-24 00:24:27 +00:00
|
|
|
|
case 'c':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_core_anal_refs (core, r_num_math (core->num, input+2), 1);
|
2010-03-24 00:24:27 +00:00
|
|
|
|
break;
|
2010-03-15 18:47:26 +00:00
|
|
|
|
case 'l':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_core_anal_graph (core, r_num_math (core->num, input+2), R_CORE_ANAL_GRAPHLINES);
|
2010-03-15 18:47:26 +00:00
|
|
|
|
break;
|
2010-03-03 13:35:18 +00:00
|
|
|
|
case 'f':
|
|
|
|
|
{
|
2010-03-15 18:47:26 +00:00
|
|
|
|
char *fname = r_str_word_get_first (input+(input[2]=='l'?3:2));
|
|
|
|
|
r_core_anal_graph_fcn (core, fname, input[2]=='l');
|
2010-03-03 13:35:18 +00:00
|
|
|
|
free (fname);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2010-03-19 03:32:42 +00:00
|
|
|
|
case 'a':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_core_anal_graph (core, r_num_math (core->num, input+2), 0);
|
2010-03-19 03:32:42 +00:00
|
|
|
|
break;
|
2010-03-03 13:35:18 +00:00
|
|
|
|
case '?':
|
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Usage: ag[?f]\n"
|
|
|
|
|
" ag [addr] ; Output graphviz code (bb at addr and childs)\n"
|
2010-03-19 03:32:42 +00:00
|
|
|
|
" aga [addr] ; Idem, but only addresses\n"
|
2010-03-24 00:24:27 +00:00
|
|
|
|
" agc [addr] ; Output graphviz call graph of function\n"
|
2010-03-15 18:47:26 +00:00
|
|
|
|
" agl [fcn name] ; Output graphviz code using meta-data\n"
|
|
|
|
|
" agf [fcn name] ; Output graphviz code of function\n"
|
|
|
|
|
" agfl [fcn name] ; Output graphviz code of function using meta-data\n");
|
2010-03-03 13:35:18 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_core_anal_graph (core, r_num_math (core->num, input+2), R_CORE_ANAL_GRAPHBODY);
|
2010-03-03 13:35:18 +00:00
|
|
|
|
}
|
2010-02-27 10:56:41 +00:00
|
|
|
|
break;
|
2010-03-04 00:46:25 +00:00
|
|
|
|
case 's':
|
|
|
|
|
switch (input[1]) {
|
|
|
|
|
case 'l':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_syscall_list (core->syscall);
|
2010-03-04 00:46:25 +00:00
|
|
|
|
break;
|
|
|
|
|
case '\0': {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
int a0 = (int)r_debug_reg_get (core->dbg, "oeax"); //XXX
|
2010-03-04 00:46:25 +00:00
|
|
|
|
cmd_syscall_do (core, a0);
|
|
|
|
|
} break;
|
|
|
|
|
case ' ':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
cmd_syscall_do (core, (int)r_num_get (core->num, input+2));
|
2010-03-04 00:46:25 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
case '?':
|
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Usage: as[?]\n"
|
|
|
|
|
" as Display syscall and arguments\n"
|
|
|
|
|
" as 4 Show syscall 4 based on asm.os\n"
|
|
|
|
|
" asl List of syscalls by asm.os and asm.arch\n");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2009-02-15 23:57:03 +00:00
|
|
|
|
default:
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
2010-03-03 02:12:24 +00:00
|
|
|
|
"Usage: a[?hobfg]\n"
|
2010-02-01 10:55:56 +00:00
|
|
|
|
" ah [handle] ; Use this analysis plugin\n"
|
2010-03-04 00:46:25 +00:00
|
|
|
|
" as [num] ; Analyze syscall using dbg.reg\n"
|
2010-02-26 12:08:42 +00:00
|
|
|
|
" ao [len] ; Analyze raw bytes\n"
|
2010-03-03 13:35:18 +00:00
|
|
|
|
" ab[?+-l*] ; Analyze basic blocks\n"
|
|
|
|
|
" af[?+-l*] ; Analyze functions\n"
|
|
|
|
|
" ag[?f] ; Output graphviz code\n");
|
2009-02-15 23:57:03 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (tbs != core->blocksize)
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_core_block_size (core, tbs);
|
2009-02-15 23:57:03 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-02-05 21:08:46 +00:00
|
|
|
|
/* TODO: simplify using r_write */
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_write(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
int i, len = strlen (input);
|
|
|
|
|
char *tmp, *str = alloca (len)+1;
|
|
|
|
|
RCore *core = (RCore *)data;
|
|
|
|
|
memcpy (str, input+1, len);
|
|
|
|
|
switch (input[0]) {
|
2010-04-08 22:52:38 +00:00
|
|
|
|
case 'A':
|
|
|
|
|
switch (input[1]) {
|
|
|
|
|
case ' ':
|
|
|
|
|
if (input[2]&&input[3]==' ') {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_asm_set_pc (core->assembler, core->offset);
|
2010-04-08 22:52:38 +00:00
|
|
|
|
eprintf ("modify (%c)=%s\n", input[2], input+4);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
len = r_asm_modify (core->assembler, core->block, input[2],
|
|
|
|
|
r_num_math (core->num, input+4));
|
2010-04-08 22:52:38 +00:00
|
|
|
|
eprintf ("len=%d\n", len);
|
|
|
|
|
if (len>0)
|
|
|
|
|
r_core_write_at (core, core->offset, core->block, len);
|
|
|
|
|
else eprintf ("r_asm_modify = %d\n", len);
|
|
|
|
|
} else eprintf ("Usage: wA [type] [value]\n");
|
|
|
|
|
break;
|
|
|
|
|
case '?':
|
|
|
|
|
default:
|
|
|
|
|
r_cons_printf ("Usage: wA [type] [value]\n"
|
|
|
|
|
"Types:\n"
|
|
|
|
|
" r raw write value\n"
|
|
|
|
|
" v set value (taking care of current address)\n"
|
|
|
|
|
" d destination register\n"
|
|
|
|
|
" 0 1st src register \n"
|
|
|
|
|
" 1 2nd src register\n"
|
|
|
|
|
"Example: wA r 0 # e800000000\n");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2010-03-30 21:12:19 +00:00
|
|
|
|
case 'c':
|
|
|
|
|
switch (input[1]) {
|
|
|
|
|
case 'i':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_cache_commit (core->io);
|
2010-03-30 21:12:19 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'r':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_cache_reset (core->io, R_TRUE);
|
2010-03-30 21:12:19 +00:00
|
|
|
|
break;
|
|
|
|
|
case '?':
|
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Usage: wc[ir*?]\n"
|
|
|
|
|
" wc list all write changes\n"
|
|
|
|
|
" wc* \"\" in radare commands\n"
|
|
|
|
|
" wcr reset all write changes in cache\n"
|
|
|
|
|
" wci commit write cache\n"
|
|
|
|
|
"NOTE: Needs io.cache=true\n");
|
|
|
|
|
break;
|
|
|
|
|
case '*':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_cache_list (core->io, R_TRUE);
|
2010-03-30 21:12:19 +00:00
|
|
|
|
break;
|
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_cache_list (core->io, R_FALSE);
|
2010-03-30 21:12:19 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case ' ':
|
2009-04-15 18:24:19 +00:00
|
|
|
|
/* write string */
|
2010-02-28 21:58:21 +00:00
|
|
|
|
len = r_str_escape (str);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd (core->io, core->file->fd);
|
|
|
|
|
r_io_write_at (core->io, core->offset, (const ut8*)str, len);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_core_block_read (core, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
case 't': {
|
2009-04-15 18:24:19 +00:00
|
|
|
|
/* TODO: Support user defined size? */
|
|
|
|
|
int len = core->blocksize;
|
2010-03-26 16:01:25 +00:00
|
|
|
|
const char *arg = (const char *)(input+((input[1]==' ')?2:1));
|
2009-09-24 10:29:05 +00:00
|
|
|
|
const ut8 *buf = core->block;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_file_dump (arg, buf, len);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
} break;
|
2009-04-15 18:24:19 +00:00
|
|
|
|
case 'T':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("TODO\n");
|
2009-04-15 18:24:19 +00:00
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
case 'f': {
|
2009-04-15 18:24:19 +00:00
|
|
|
|
int size;
|
2010-03-26 16:01:25 +00:00
|
|
|
|
const char *arg = (const char *)(input+((input[1]==' ')?2:1));
|
2010-02-28 21:58:21 +00:00
|
|
|
|
ut8 *buf = (ut8*) r_file_slurp (arg, &size);
|
2009-09-25 02:04:51 +00:00
|
|
|
|
if (buf) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd (core->io, core->file->fd);
|
|
|
|
|
r_io_write_at (core->io, core->offset, buf, size);
|
2009-04-15 18:24:19 +00:00
|
|
|
|
free(buf);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
} else eprintf ("Cannot open file '%s'\n", arg);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
} break;
|
|
|
|
|
case 'F': {
|
2009-04-15 18:24:19 +00:00
|
|
|
|
int size;
|
2010-03-26 16:01:25 +00:00
|
|
|
|
const char *arg = (const char *)(input+((input[1]==' ')?2:1));
|
2010-02-28 21:58:21 +00:00
|
|
|
|
ut8 *buf = r_file_slurp_hexpairs (arg, &size);
|
2009-04-15 18:24:19 +00:00
|
|
|
|
if (buf == NULL) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd (core->io, core->file->fd);
|
|
|
|
|
r_io_write_at (core->io, core->offset, buf, size);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
free (buf);
|
|
|
|
|
} else eprintf ("Cannot open file '%s'\n", arg);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
} break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 'w':
|
|
|
|
|
str = str+1;
|
|
|
|
|
len = len-1;
|
|
|
|
|
len *= 2;
|
|
|
|
|
tmp = alloca(len);
|
2010-02-01 10:55:56 +00:00
|
|
|
|
for (i=0;i<len;i++) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (i%2) tmp[i] = 0;
|
|
|
|
|
else tmp[i] = str[i>>1];
|
|
|
|
|
}
|
|
|
|
|
str = tmp;
|
|
|
|
|
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd (core->io, core->file->fd);
|
|
|
|
|
r_io_write_at (core->io, core->offset, (const ut8*)str, len);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_core_block_read (core, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'x':
|
|
|
|
|
{
|
2010-02-28 21:58:21 +00:00
|
|
|
|
int len = strlen (input);
|
|
|
|
|
ut8 *buf = alloca (len);
|
|
|
|
|
len = r_hex_str2bin (input+1, buf);
|
|
|
|
|
r_core_write_at (core, core->offset, buf, len);
|
|
|
|
|
r_core_block_read (core, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
2009-04-07 00:26:41 +00:00
|
|
|
|
case 'a':
|
|
|
|
|
{
|
2010-04-14 22:29:42 +00:00
|
|
|
|
RAsmCode *acode;
|
2009-04-08 23:03:49 +00:00
|
|
|
|
/* XXX ULTRAUGLY , needs fallback support in rasm */
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_asm_use (core->assembler, "x86.olly");
|
|
|
|
|
r_asm_set_pc (core->assembler, core->offset);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (input[1]==' ') input=input+1;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
acode = r_asm_massemble (core->assembler, input+1);
|
2010-04-14 22:29:42 +00:00
|
|
|
|
if (acode) {
|
|
|
|
|
eprintf ("Written %d bytes (%s)=wx %s\n", acode->len, input+1, acode->buf_hex);
|
|
|
|
|
r_core_write_at (core, core->offset, acode->buf, acode->len);
|
|
|
|
|
r_asm_code_free (acode);
|
|
|
|
|
r_core_block_read (core, 0);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_asm_use (core->assembler, "x86"); /* XXX */
|
2010-04-14 22:29:42 +00:00
|
|
|
|
}
|
2009-04-07 00:26:41 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 'b':
|
|
|
|
|
{
|
2010-02-28 21:58:21 +00:00
|
|
|
|
int len = strlen (input);
|
|
|
|
|
ut8 *buf = alloca (len);
|
|
|
|
|
len = r_hex_str2bin (input+1, buf);
|
|
|
|
|
r_mem_copyloop (core->block, buf, core->blocksize, len);
|
|
|
|
|
r_core_write_at (core, core->offset, core->block, core->blocksize);
|
|
|
|
|
r_core_block_read (core, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'm':
|
|
|
|
|
{
|
2010-02-28 21:58:21 +00:00
|
|
|
|
int len = r_hex_str2bin (input+1, (ut8*)str);
|
|
|
|
|
switch (input[1]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '\0':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("Current write mask: TODO\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
// TODO
|
|
|
|
|
break;
|
|
|
|
|
case '?':
|
|
|
|
|
break;
|
|
|
|
|
case '-':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_write_mask(core->io, 0, 0);
|
2010-02-22 03:25:35 +00:00
|
|
|
|
eprintf ("Write mask disabled\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case ' ':
|
|
|
|
|
if (len == 0) {
|
2010-02-01 10:55:56 +00:00
|
|
|
|
eprintf ("Invalid string\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
} else {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd(core->io, core->file->fd);
|
|
|
|
|
r_io_set_write_mask(core->io, (const ut8*)str, len);
|
2010-02-01 10:55:56 +00:00
|
|
|
|
eprintf ("Write mask set to '");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
for (i=0;i<len;i++)
|
2010-02-01 10:55:56 +00:00
|
|
|
|
eprintf ("%02x", str[i]);
|
|
|
|
|
eprintf ("'\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'v':
|
|
|
|
|
{
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ut64 off = r_num_math(core->num, input+1);
|
|
|
|
|
r_io_set_fd (core->io, core->file->fd);
|
|
|
|
|
r_io_seek (core->io, core->offset, R_IO_SEEK_SET);
|
2009-07-16 12:17:32 +00:00
|
|
|
|
if (off&UT64_32U) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
/* 8 byte addr */
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut64 addr8;
|
|
|
|
|
memcpy((ut8*)&addr8, (ut8*)&off, 8); // XXX needs endian here
|
|
|
|
|
// endian_memcpy((ut8*)&addr8, (ut8*)&off, 8);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_write(core->io, (const ut8 *)&addr8, 8);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
} else {
|
|
|
|
|
/* 4 byte addr */
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut32 addr4, addr4_ = (ut32)off;
|
|
|
|
|
//drop_endian((ut8*)&addr4_, (ut8*)&addr4, 4); /* addr4_ = addr4 */
|
|
|
|
|
//endian_memcpy((ut8*)&addr4, (ut8*)&addr4_, 4); /* addr4 = addr4_ */
|
2010-02-28 21:58:21 +00:00
|
|
|
|
memcpy ((ut8*)&addr4, (ut8*)&addr4_, 4); // XXX needs endian here too
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_write (core->io, (const ut8 *)&addr4, 4);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_core_block_read (core, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'o':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
switch (input[1]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 'a':
|
|
|
|
|
case 's':
|
|
|
|
|
case 'A':
|
|
|
|
|
case 'x':
|
|
|
|
|
case 'r':
|
|
|
|
|
case 'l':
|
|
|
|
|
case 'm':
|
|
|
|
|
case 'd':
|
|
|
|
|
case 'o':
|
|
|
|
|
if (input[2]!=' ') {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("Usage: 'wo%c 00 11 22'\n", input[1]);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
case '2':
|
|
|
|
|
case '4':
|
|
|
|
|
r_core_write_op(core, input+3, input[1]);
|
|
|
|
|
break;
|
|
|
|
|
case '\0':
|
|
|
|
|
case '?':
|
|
|
|
|
default:
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: wo[xrlasmd] [hexpairs]\n"
|
|
|
|
|
"Example: wox 90 ; xor cur block with 90\n"
|
|
|
|
|
"Example: woa 02 03 ; add 2, 3 to all bytes of cur block\n"
|
|
|
|
|
"Supported operations:\n"
|
|
|
|
|
" woa addition +=\n"
|
|
|
|
|
" wos substraction -=\n"
|
|
|
|
|
" wom multiply *=\n"
|
|
|
|
|
" wod divide /=\n"
|
|
|
|
|
" wox xor ^=\n"
|
|
|
|
|
" woo or |=\n"
|
|
|
|
|
" woA and &=\n"
|
|
|
|
|
" wor shift right >>=\n"
|
|
|
|
|
" wol shift left <<=\n"
|
|
|
|
|
" wo2 2 byte endian swap 2=\n"
|
|
|
|
|
" wo4 4 byte endian swap 4=\n"
|
|
|
|
|
);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
case '?':
|
|
|
|
|
if (core->oobi) {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("Writing oobi buffer!\n");
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd (core->io, core->file->fd);
|
|
|
|
|
r_io_write (core->io, core->oobi, core->oobi_len);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
} else
|
2010-04-08 22:52:38 +00:00
|
|
|
|
r_cons_printf (
|
2010-02-01 10:55:56 +00:00
|
|
|
|
"Usage: w[x] [str] [<file] [<<EOF] [@addr]\n"
|
2010-03-30 21:12:19 +00:00
|
|
|
|
" w foobar write string 'foobar'\n"
|
|
|
|
|
" ww foobar write wide string 'f\\x00o\\x00o\\x00b\\x00a\\x00r\\x00'\n"
|
|
|
|
|
" wa push ebp write opcode, separated by ';' (use '\"' around the command)\n"
|
2010-04-08 22:52:38 +00:00
|
|
|
|
" wA r 0 alter/modify opcode at current seek (see wA?)\n"
|
2010-03-30 21:12:19 +00:00
|
|
|
|
" wb 010203 fill current block with cyclic hexpairs\n"
|
|
|
|
|
" wc[ir*?] write cache commit/reset/list\n"
|
|
|
|
|
" wx 9090 write two intel nops\n"
|
|
|
|
|
" wv eip+34 write 32-64 bit value\n"
|
|
|
|
|
" wo[] hex write in block with operation. 'wo?' fmi\n"
|
2010-04-08 22:52:38 +00:00
|
|
|
|
" wm f0ff set binary mask hexpair to be used as cyclic write mask\n"
|
2010-03-30 21:12:19 +00:00
|
|
|
|
" wf file write contents of file at current offset\n"
|
|
|
|
|
" wF file write contents of hexpairs file here\n"
|
|
|
|
|
" wt file write current block to file\n");
|
2009-04-15 18:24:19 +00:00
|
|
|
|
//TODO: add support for offset+seek
|
|
|
|
|
// " wf file o s ; write contents of file from optional offset 'o' and size 's'.\n"
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static const char *cmdhit = NULL;
|
2010-04-06 16:21:41 +00:00
|
|
|
|
static int __cb_hit(RSearchKeyword *kw, void *user, ut64 addr) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)user;
|
2009-04-02 10:23:32 +00:00
|
|
|
|
|
2010-04-14 11:02:23 +00:00
|
|
|
|
r_cons_printf ("f hit%d_%d %d 0x%08"PFMT64x"\n",
|
2009-04-02 10:23:32 +00:00
|
|
|
|
kw->kwidx, kw->count, kw->keyword_length, addr);
|
|
|
|
|
|
2010-02-01 10:55:56 +00:00
|
|
|
|
if (!strnull (cmdhit)) {
|
2010-01-12 01:12:18 +00:00
|
|
|
|
ut64 here = core->offset;
|
2010-02-21 10:35:49 +00:00
|
|
|
|
r_core_seek (core, addr, R_FALSE);
|
2010-03-19 03:02:23 +00:00
|
|
|
|
r_core_cmd (core, cmdhit, 0);
|
2010-02-21 10:35:49 +00:00
|
|
|
|
r_core_seek (core, here, R_TRUE);
|
2009-04-02 10:23:32 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-02-09 00:54:09 +00:00
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_search(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut64 at;
|
|
|
|
|
ut32 n32;
|
2009-02-09 00:54:09 +00:00
|
|
|
|
int ret, dosearch = 0;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut8 *buf;
|
2009-02-09 00:54:09 +00:00
|
|
|
|
switch (input[0]) {
|
|
|
|
|
case '/':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_search_begin (core->search);
|
2009-02-09 00:54:09 +00:00
|
|
|
|
dosearch = 1;
|
|
|
|
|
break;
|
|
|
|
|
case 'v':
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_search_free (core->search);
|
|
|
|
|
core->search = r_search_new (R_SEARCH_KEYWORD);
|
2010-04-08 22:52:38 +00:00
|
|
|
|
r_search_set_distance (core->search, (int)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_get_i (core->config, "search.distance"));
|
|
|
|
|
n32 = r_num_math (core->num, input+1);
|
2010-04-06 11:02:21 +00:00
|
|
|
|
r_search_kw_add (core->search,
|
|
|
|
|
r_search_keyword_new ((const ut8*)&n32, 4, NULL, 0, NULL));
|
2010-02-21 10:35:49 +00:00
|
|
|
|
r_search_begin (core->search);
|
2009-02-09 00:54:09 +00:00
|
|
|
|
dosearch = 1;
|
|
|
|
|
break;
|
|
|
|
|
case ' ': /* search string */
|
|
|
|
|
r_search_free(core->search);
|
2010-04-06 11:02:21 +00:00
|
|
|
|
core->search = r_search_new (R_SEARCH_KEYWORD);
|
2010-04-08 22:52:38 +00:00
|
|
|
|
r_search_set_distance (core->search, (int)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_get_i (core->config, "search.distance"));
|
2010-04-06 11:02:21 +00:00
|
|
|
|
r_search_kw_add (core->search,
|
|
|
|
|
r_search_keyword_new_str (input+1, "", NULL));
|
|
|
|
|
r_search_begin (core->search);
|
2009-02-09 00:54:09 +00:00
|
|
|
|
dosearch = 1;
|
|
|
|
|
break;
|
2009-02-16 23:09:40 +00:00
|
|
|
|
case 'm': /* match regexp */
|
|
|
|
|
{
|
2010-03-30 21:12:19 +00:00
|
|
|
|
char *inp = strdup (input+2);
|
|
|
|
|
char *res = r_str_lchr (inp+1, inp[0]);
|
2009-02-16 23:09:40 +00:00
|
|
|
|
char *opt = NULL;
|
|
|
|
|
if (res > inp) {
|
|
|
|
|
opt = strdup(res+1);
|
|
|
|
|
res[1]='\0';
|
|
|
|
|
}
|
2010-03-30 21:12:19 +00:00
|
|
|
|
r_search_free (core->search);
|
|
|
|
|
core->search = r_search_new (R_SEARCH_REGEXP);
|
2010-04-08 22:52:38 +00:00
|
|
|
|
r_search_set_distance (core->search, (int)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_get_i (core->config, "search.distance"));
|
2010-04-06 11:02:21 +00:00
|
|
|
|
r_search_kw_add (core->search,
|
|
|
|
|
r_search_keyword_new_str (inp, opt, NULL));
|
2010-03-30 21:12:19 +00:00
|
|
|
|
r_search_begin (core->search);
|
2009-02-16 23:09:40 +00:00
|
|
|
|
dosearch = 1;
|
|
|
|
|
free(inp);
|
|
|
|
|
free(opt);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2009-02-09 00:54:09 +00:00
|
|
|
|
case 'x': /* search hex */
|
2010-04-08 16:01:45 +00:00
|
|
|
|
r_search_free (core->search);
|
|
|
|
|
core->search = r_search_new (R_SEARCH_KEYWORD);
|
2010-04-08 22:52:38 +00:00
|
|
|
|
r_search_set_distance (core->search, (int)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_get_i (core->config, "search.distance"));
|
2010-04-06 11:02:21 +00:00
|
|
|
|
r_search_kw_add (core->search,
|
2010-04-08 16:01:45 +00:00
|
|
|
|
r_search_keyword_new_hexmask (input+2, NULL));
|
|
|
|
|
r_search_begin (core->search);
|
2009-02-09 00:54:09 +00:00
|
|
|
|
dosearch = 1;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Usage: /[xm/] [arg]\n"
|
2009-02-09 00:54:09 +00:00
|
|
|
|
" / foo ; search for string 'foo'\n"
|
2009-02-16 23:09:40 +00:00
|
|
|
|
" /m /E.F/i ; match regular expression\n"
|
2009-02-09 00:54:09 +00:00
|
|
|
|
" /x ff0033 ; search for hex string\n"
|
|
|
|
|
" // ; repeat last search\n");
|
|
|
|
|
break;
|
|
|
|
|
}
|
2009-02-16 23:09:40 +00:00
|
|
|
|
if (core->search->n_kws==0) {
|
2010-02-01 10:55:56 +00:00
|
|
|
|
eprintf ("No keywords defined\n");
|
2009-02-16 23:09:40 +00:00
|
|
|
|
} else
|
2009-02-09 00:54:09 +00:00
|
|
|
|
if (dosearch) {
|
|
|
|
|
/* set callback */
|
|
|
|
|
/* TODO: handle last block of data */
|
|
|
|
|
/* TODO: handle ^C */
|
|
|
|
|
/* TODO: launch search in background support */
|
2009-07-08 11:49:55 +00:00
|
|
|
|
buf = (ut8 *)malloc(core->blocksize);
|
2009-04-02 10:23:32 +00:00
|
|
|
|
r_search_set_callback(core->search, &__cb_hit, core);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
cmdhit = r_config_get(core->config, "cmd.hit");
|
2009-02-16 23:09:40 +00:00
|
|
|
|
r_cons_break(NULL, NULL);
|
2010-01-12 01:12:18 +00:00
|
|
|
|
for(at = core->offset; at < core->file->size; at += core->blocksize) {
|
2010-02-18 15:36:55 +00:00
|
|
|
|
if (r_cons_singleton()->breaked)
|
2009-02-16 23:09:40 +00:00
|
|
|
|
break;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd(core->io, core->file->fd);
|
|
|
|
|
ret = r_io_read_at(core->io, at, buf, core->blocksize);
|
2009-02-09 00:54:09 +00:00
|
|
|
|
if (ret != core->blocksize)
|
|
|
|
|
break;
|
2009-02-16 23:09:40 +00:00
|
|
|
|
if (r_search_update(core->search, &at, buf, ret) == -1) {
|
|
|
|
|
printf("search:update error\n");
|
|
|
|
|
break;
|
|
|
|
|
}
|
2009-02-09 00:54:09 +00:00
|
|
|
|
}
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_cons_break_end ();
|
2009-02-09 00:54:09 +00:00
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_eval(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-02-01 10:55:56 +00:00
|
|
|
|
switch (input[0]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_list (core->config, NULL, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-04-06 22:53:25 +00:00
|
|
|
|
case '!':
|
|
|
|
|
input = r_str_chop_ro(input+1);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (!r_config_swap (core->config, input))
|
2010-02-01 10:55:56 +00:00
|
|
|
|
eprintf ("r_config: '%s' is not a boolean variable.\n", input);
|
2009-04-06 22:53:25 +00:00
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case '-':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_core_config_init (core);
|
|
|
|
|
eprintf ("BUG: 'e-' command locks the eval hashtable. patches are welcome :)\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '*':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_list (core->config, NULL, 1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '?':
|
2010-02-02 10:09:52 +00:00
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: e[?] [var[=value]]\n"
|
|
|
|
|
" e ; list config vars\n"
|
|
|
|
|
" e- ; reset config vars\n"
|
|
|
|
|
" e* ; dump config vars in r commands\n"
|
2009-04-06 22:53:25 +00:00
|
|
|
|
" e!a ; invert the boolean value of 'a' var\n"
|
2009-02-05 21:08:46 +00:00
|
|
|
|
" e a ; get value of var 'a'\n"
|
|
|
|
|
" e a=b ; set var 'a' the 'b' value\n");
|
2010-05-19 22:59:42 +00:00
|
|
|
|
//r_cmd_help(core->cmd, "e");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_config_eval (core->config, input);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_hash(void *data, const char *input) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
char algo[32];
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut32 len = core->blocksize;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
const char *ptr;
|
|
|
|
|
|
|
|
|
|
if (input[0]=='!') {
|
|
|
|
|
#if 0
|
|
|
|
|
#!lua < file
|
|
|
|
|
#!lua <<EOF
|
|
|
|
|
#!lua
|
|
|
|
|
#!lua foo bar
|
|
|
|
|
#endif
|
2009-02-10 23:56:20 +00:00
|
|
|
|
if (input[1]=='\0') {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_lang_list (core->lang);
|
2009-02-10 23:56:20 +00:00
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
// TODO: set argv here
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_lang_use (core->lang, input+1);
|
|
|
|
|
r_lang_setup (core->lang);
|
2009-02-10 23:56:20 +00:00
|
|
|
|
if (core->oobi)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_lang_run (core->lang,(const char *)
|
2009-04-02 10:23:32 +00:00
|
|
|
|
core->oobi, core->oobi_len);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
else r_lang_prompt (core->lang);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-02 10:09:52 +00:00
|
|
|
|
ptr = strchr (input, ' ');
|
|
|
|
|
sscanf (input, "%31s", algo);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (ptr != NULL)
|
2010-05-19 22:59:42 +00:00
|
|
|
|
len = r_num_math (core->num, ptr+1);
|
2010-02-02 10:09:52 +00:00
|
|
|
|
/* TODO: support all hash algorithms and so */
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (!r_str_ccmp(input, "crc32", ' ')) {
|
2010-03-26 16:01:25 +00:00
|
|
|
|
r_cons_printf ("%04x\n", r_hash_crc32 (core->block, len));
|
2009-02-05 21:08:46 +00:00
|
|
|
|
} else
|
|
|
|
|
if (!r_str_ccmp(input, "crc16", ' ')) {
|
2010-03-26 16:01:25 +00:00
|
|
|
|
r_cons_printf ("%02x\n", r_hash_crc16 (0, core->block, len));
|
|
|
|
|
} else
|
|
|
|
|
if (input[0]=='?') {
|
|
|
|
|
r_cons_printf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: #algo <size> @ addr\n"
|
|
|
|
|
" #crc32 ; calculate crc32 of current block\n"
|
|
|
|
|
" #crc32 < /etc/fstab ; calculate crc32 of this file\n"
|
|
|
|
|
" #md5 128K @ edi ; calculate md5 of 128K from 'edi'\n"
|
|
|
|
|
"Usage #!interpreter [<args>] [<file] [<<eof]\n"
|
|
|
|
|
" #! ; list all available interpreters\n"
|
|
|
|
|
" #!python ; run python commandline\n"
|
|
|
|
|
" #!python < foo.py ; run foo.py python script\n"
|
|
|
|
|
" #!python <<EOF ; get python code until 'EOF' mark\n"
|
2010-02-02 10:09:52 +00:00
|
|
|
|
" #!python arg0 a1 <<q ; set arg0 and arg1 and read until 'q'\n"
|
|
|
|
|
"Comments:\n"
|
|
|
|
|
" # this is a comment ; note the space after the sharp sign\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_visual(void *data, const char *input) {
|
2010-02-01 10:55:56 +00:00
|
|
|
|
return r_core_visual ((RCore *)data, input);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_system(void *data, const char *input) {
|
2010-02-02 10:09:52 +00:00
|
|
|
|
return r_sys_cmd (input);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_open(void *data, const char *input) {
|
2010-02-01 10:55:56 +00:00
|
|
|
|
RCore *core = (RCore*)data;
|
|
|
|
|
RCoreFile *file;
|
|
|
|
|
ut64 addr, size;
|
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
|
|
switch (*input) {
|
|
|
|
|
case '\0':
|
|
|
|
|
r_core_file_list (core);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
case '?':
|
2010-03-01 09:49:04 +00:00
|
|
|
|
eprintf ("Usage: o [file] ([offset])\n"
|
|
|
|
|
" o ; list opened files\n"
|
|
|
|
|
" o /bin/ls ; open /bin/ls file\n"
|
|
|
|
|
" o /bin/ls 0x8048000 ; map file\n"
|
|
|
|
|
" o-1 ; close file index 1\n");
|
2010-02-01 10:55:56 +00:00
|
|
|
|
break;
|
|
|
|
|
case ' ':
|
|
|
|
|
ptr = strchr (input+1, ' ');
|
|
|
|
|
if (ptr)
|
|
|
|
|
*ptr = '\0';
|
|
|
|
|
file = r_core_file_open (core, input+1, R_IO_READ);
|
|
|
|
|
if (file) {
|
|
|
|
|
if (ptr) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
addr = r_num_math (core->num, ptr+1);
|
|
|
|
|
size = r_io_size (core->io, file->fd);
|
|
|
|
|
r_io_map_add (core->io, file->fd, R_IO_READ, 0, addr, size);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("Map '%s' in 0x%08"PFMT64x" with size 0x%"PFMT64x"\n",
|
2010-02-01 10:55:56 +00:00
|
|
|
|
input+1, addr, size);
|
|
|
|
|
}
|
|
|
|
|
} else eprintf ("Cannot open file '%s'\n", input+1);
|
|
|
|
|
break;
|
|
|
|
|
case '-':
|
|
|
|
|
file = r_core_file_get_fd (core, atoi (input+1));
|
|
|
|
|
if (file)
|
|
|
|
|
r_core_file_close (core, file);
|
|
|
|
|
else eprintf ("Unable to find filedescriptor %d\n", atoi (input+1));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_meta(void *data, const char *input) {
|
|
|
|
|
RCore *core = (RCore*)data;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
int i, ret, line = 0;
|
2010-02-22 19:10:52 +00:00
|
|
|
|
char file[1024];
|
2010-02-28 21:58:21 +00:00
|
|
|
|
switch (input[0]) {
|
2009-04-06 22:53:25 +00:00
|
|
|
|
case '*':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_meta_list (core->meta, R_META_ANY);
|
2009-02-16 10:24:45 +00:00
|
|
|
|
break;
|
2009-03-27 09:47:19 +00:00
|
|
|
|
case 'L': // debug information of current offset
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ret = r_bin_meta_get_line (core->bin, core->offset, file, 1023, &line);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (ret) {
|
|
|
|
|
r_cons_printf ("file %s\nline %d\n", file, line);
|
|
|
|
|
ret = (line<5)? 5-line: 5;
|
|
|
|
|
line -= 2;
|
|
|
|
|
for (i = 0; i<ret; i++) {
|
|
|
|
|
char *row = r_file_slurp_line (file, line+i, 0);
|
|
|
|
|
r_cons_printf ("%c %.3x %s\n", (i==2)?'>':' ', line+i, row);
|
|
|
|
|
}
|
2010-04-14 11:02:23 +00:00
|
|
|
|
} else eprintf ("Cannot find meta information at 0x%08"PFMT64x"\n", core->offset);
|
2009-03-27 00:16:27 +00:00
|
|
|
|
break;
|
2009-02-16 10:24:45 +00:00
|
|
|
|
case 'C': /* add comment */
|
2009-04-06 22:53:25 +00:00
|
|
|
|
// TODO: do we need to get the size? or the offset?
|
|
|
|
|
// TODO: is this an exception compared to other C? commands?
|
|
|
|
|
if (input[1]==' ') input = input+1;
|
2010-02-01 10:55:56 +00:00
|
|
|
|
if (input[1]=='-')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_meta_del (core->meta, R_META_COMMENT, core->offset, 1, input+2);
|
|
|
|
|
else r_meta_add (core->meta, R_META_COMMENT, core->offset, 1, input+1);
|
2009-02-16 10:24:45 +00:00
|
|
|
|
break;
|
2009-04-06 22:53:25 +00:00
|
|
|
|
case 'S':
|
|
|
|
|
case 's':
|
|
|
|
|
case 'm': /* struct */
|
|
|
|
|
case 'x': /* code xref */
|
|
|
|
|
case 'X': /* data xref */
|
2009-02-16 10:24:45 +00:00
|
|
|
|
case 'F': /* add function */
|
2009-04-06 22:53:25 +00:00
|
|
|
|
{
|
2010-01-12 01:12:18 +00:00
|
|
|
|
ut64 addr = core->offset;
|
2009-04-06 22:53:25 +00:00
|
|
|
|
char fun_name[128];
|
|
|
|
|
int size = atoi(input);
|
|
|
|
|
int type = R_META_FUNCTION;
|
2010-02-22 23:26:13 +00:00
|
|
|
|
char *t, *p = strchr (input+1, ' ');
|
2009-04-06 22:53:25 +00:00
|
|
|
|
if (p) {
|
2010-02-22 23:26:13 +00:00
|
|
|
|
t = strdup (p+1);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
//eprintf ("T=(%s)\n", t);
|
2010-03-01 09:49:04 +00:00
|
|
|
|
p = strchr (t, ' ');
|
2009-04-06 22:53:25 +00:00
|
|
|
|
if (p) {
|
|
|
|
|
*p='\0';
|
2010-02-22 23:26:13 +00:00
|
|
|
|
strncpy (fun_name, p+1, sizeof (fun_name));
|
2010-04-14 11:02:23 +00:00
|
|
|
|
} else sprintf (fun_name, "sub_%08"PFMT64x"", addr);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
addr = r_num_math (core->num, t);
|
2009-04-06 22:53:25 +00:00
|
|
|
|
free(t);
|
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_meta_add (core->meta, type, addr, size, fun_name);
|
2009-04-06 22:53:25 +00:00
|
|
|
|
}
|
2009-02-16 10:24:45 +00:00
|
|
|
|
break;
|
2009-04-06 22:53:25 +00:00
|
|
|
|
case '\0':
|
2009-03-27 00:16:27 +00:00
|
|
|
|
case '?':
|
2010-03-01 09:49:04 +00:00
|
|
|
|
eprintf (
|
2009-03-27 00:16:27 +00:00
|
|
|
|
"Usage: C[CDF?] [arg]\n"
|
2009-04-06 22:53:25 +00:00
|
|
|
|
" CL [addr] ; show 'code line' information (bininfo)\n"
|
|
|
|
|
" CF [size] [name] [addr] [name] ; register function size here (TODO)\n"
|
|
|
|
|
" CC [string] ; add comment (TODO)\n");
|
2009-02-16 10:24:45 +00:00
|
|
|
|
}
|
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_io_system(void *data, const char *input) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_io_set_fd(core->io, core->file->fd);
|
|
|
|
|
return r_io_system(core->io, input);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
static int cmd_macro(void *data, const char *input) {
|
|
|
|
|
RCore *core = (RCore*)data;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
switch(input[0]) {
|
|
|
|
|
case ')':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cmd_macro_break (&core->cmd->macro, input+1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '-':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cmd_macro_rm (&core->cmd->macro, input+1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '\0':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cmd_macro_list (&core->cmd->macro);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case '?':
|
2010-02-22 23:26:13 +00:00
|
|
|
|
eprintf (
|
2009-02-05 21:08:46 +00:00
|
|
|
|
"Usage: (foo\\n..cmds..\\n)\n"
|
|
|
|
|
" Record macros grouping commands\n"
|
|
|
|
|
" (foo args\\n ..) ; define a macro\n"
|
|
|
|
|
" (-foo) ; remove a macro\n"
|
|
|
|
|
" .(foo) ; to call it\n"
|
|
|
|
|
" () ; break inside macro\n"
|
|
|
|
|
"Argument support:\n"
|
|
|
|
|
" (foo x y\\n$1 @ $2) ; define fun with args\n"
|
|
|
|
|
" .(foo 128 0x804800) ; call it with args\n"
|
|
|
|
|
"Iterations:\n"
|
|
|
|
|
" .(foo\\n() $@) ; define iterator returning iter index\n"
|
|
|
|
|
" x @@ .(foo) ; iterate over them\n"
|
|
|
|
|
);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cmd_macro_add (&core->cmd->macro, input);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 21:12:19 +00:00
|
|
|
|
static int r_core_cmd_pipe(RCore *core, char *radare_cmd, char *shell_cmd) {
|
2010-01-15 16:02:04 +00:00
|
|
|
|
#if __UNIX__
|
2009-12-31 00:27:03 +00:00
|
|
|
|
int fds[2];
|
2010-01-15 16:02:04 +00:00
|
|
|
|
int stdout_fd, status;
|
|
|
|
|
|
2010-03-01 09:49:04 +00:00
|
|
|
|
stdout_fd = dup (1);
|
|
|
|
|
pipe (fds);
|
|
|
|
|
radare_cmd = r_str_trim_head (radare_cmd);
|
|
|
|
|
shell_cmd = r_str_trim_head (shell_cmd);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
if (fork()) {
|
|
|
|
|
dup2(fds[1], 1);
|
2010-03-01 09:49:04 +00:00
|
|
|
|
close (fds[1]);
|
|
|
|
|
close (fds[0]);
|
|
|
|
|
r_core_cmd (core, radare_cmd, 0);
|
|
|
|
|
r_cons_flush ();
|
|
|
|
|
close (1);
|
|
|
|
|
wait (&status);
|
|
|
|
|
dup2 (stdout_fd, 1);
|
|
|
|
|
close (stdout_fd);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
} else {
|
2010-03-01 09:49:04 +00:00
|
|
|
|
close (fds[1]);
|
|
|
|
|
dup2 (fds[0], 0);
|
|
|
|
|
dup2 (2, 1);
|
|
|
|
|
execl ("/bin/sh", "sh", "-c", shell_cmd, NULL);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
}
|
|
|
|
|
return status;
|
2010-01-15 16:02:04 +00:00
|
|
|
|
#else
|
|
|
|
|
#warning r_core_cmd_pipe UNIMPLEMENTED FOR THIS PLATFORM
|
|
|
|
|
return -1;
|
|
|
|
|
#endif
|
2009-12-31 00:27:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 21:12:19 +00:00
|
|
|
|
static int r_core_cmd_subst(RCore *core, char *cmd) {
|
2009-04-15 18:24:19 +00:00
|
|
|
|
char *ptr, *ptr2, *str;
|
2009-12-31 00:27:03 +00:00
|
|
|
|
int i, len = strlen(cmd), pipefd, ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
2009-12-31 00:27:03 +00:00
|
|
|
|
if (!*cmd || cmd[0]=='\0')
|
2009-02-05 21:28:33 +00:00
|
|
|
|
return 0;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
cmd = r_str_trim_head_tail (cmd);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
|
|
|
|
|
/* quoted / raw command */
|
|
|
|
|
if (cmd[0] =='"') {
|
|
|
|
|
if (cmd[len-1] != '"') {
|
2010-02-22 23:26:13 +00:00
|
|
|
|
eprintf ("parse: Missing ending '\"'\n");
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
2010-02-22 23:26:13 +00:00
|
|
|
|
cmd[len-1]='\0';
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ret = r_cmd_call (core->cmd, cmd+1);
|
2010-02-22 23:26:13 +00:00
|
|
|
|
return ret;
|
2009-12-31 00:27:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* multiple commands */
|
2010-02-22 23:26:13 +00:00
|
|
|
|
ptr = strrchr (cmd, ';');
|
2009-12-31 00:27:03 +00:00
|
|
|
|
if (ptr) {
|
2009-02-05 21:08:46 +00:00
|
|
|
|
ptr[0]='\0';
|
2010-02-22 23:26:13 +00:00
|
|
|
|
if (r_core_cmd_subst (core, cmd) == -1)
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return -1;
|
|
|
|
|
cmd = ptr+1;
|
2010-02-22 23:26:13 +00:00
|
|
|
|
r_cons_flush ();
|
2009-12-31 00:27:03 +00:00
|
|
|
|
}
|
2009-02-18 00:43:57 +00:00
|
|
|
|
|
2009-12-31 00:27:03 +00:00
|
|
|
|
/* pipe console to shell process */
|
2010-02-22 23:26:13 +00:00
|
|
|
|
ptr = strchr (cmd, '|');
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (ptr) {
|
|
|
|
|
ptr[0] = '\0';
|
2010-02-22 23:26:13 +00:00
|
|
|
|
r_core_cmd_pipe (core, cmd, ptr+1);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* bool conditions */
|
|
|
|
|
ptr = strchr(cmd, '&');
|
|
|
|
|
while (ptr&&ptr[1]=='&') {
|
|
|
|
|
ptr[0]='\0';
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ret = r_cmd_call (core->cmd, cmd);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
if (ret == -1){
|
2010-02-22 23:26:13 +00:00
|
|
|
|
eprintf ("command error(%s)\n", cmd);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return ret;
|
|
|
|
|
}
|
2010-02-22 23:26:13 +00:00
|
|
|
|
for (cmd=ptr+2;cmd&&cmd[0]==' ';cmd=cmd+1);
|
|
|
|
|
ptr = strchr (cmd, '&');
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Out Of Band Input */
|
2010-02-22 23:26:13 +00:00
|
|
|
|
free (core->oobi);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
core->oobi = NULL;
|
2010-02-22 23:26:13 +00:00
|
|
|
|
ptr = strchr (cmd, '<');
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (ptr) {
|
|
|
|
|
ptr[0] = '\0';
|
|
|
|
|
if (ptr[1]=='<') {
|
|
|
|
|
/* this is a bit mess */
|
2010-02-28 21:58:21 +00:00
|
|
|
|
const char *oprompt = r_line_singleton ()->prompt;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
oprompt = ">";
|
2010-02-28 21:58:21 +00:00
|
|
|
|
for (str=ptr+2; str[0]==' '; str=str+1);
|
|
|
|
|
eprintf ("==> Reading from stdin until '%s'\n", str);
|
|
|
|
|
free (core->oobi);
|
|
|
|
|
core->oobi = malloc (1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
core->oobi[0] = '\0';
|
|
|
|
|
core->oobi_len = 0;
|
|
|
|
|
for (;;) {
|
|
|
|
|
char buf[1024];
|
|
|
|
|
int ret;
|
|
|
|
|
printf("> "); fflush(stdout);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
fgets(buf, 1023, stdin); // XXX use r_line ??
|
2010-03-01 09:49:04 +00:00
|
|
|
|
if (feof (stdin))
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2010-03-01 09:49:04 +00:00
|
|
|
|
buf[strlen (buf)-1]='\0';
|
|
|
|
|
ret = strlen (buf);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
core->oobi_len+=ret;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
core->oobi = realloc (core->oobi, core->oobi_len+1);
|
|
|
|
|
if (!strcmp (buf, str))
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
strcat ((char *)core->oobi, buf);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
2010-02-18 15:36:55 +00:00
|
|
|
|
r_line_singleton ()->prompt = oprompt;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
} else {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
for (str=ptr+1;str[0]== ' ';str=str+1);
|
|
|
|
|
eprintf ("SLURPING FILE '%s'\n", str);
|
|
|
|
|
core->oobi = (ut8*)r_file_slurp (str, &core->oobi_len);
|
2009-03-14 11:39:37 +00:00
|
|
|
|
if (core->oobi == NULL)
|
2010-02-05 11:21:37 +00:00
|
|
|
|
eprintf ("Cannot open file\n");
|
2009-03-14 11:39:37 +00:00
|
|
|
|
else if (ptr == cmd)
|
2010-02-05 11:21:37 +00:00
|
|
|
|
return r_core_cmd_buffer (core, (const char *)core->oobi);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2009-12-31 00:27:03 +00:00
|
|
|
|
|
|
|
|
|
/* pipe console to file */
|
2010-02-28 21:58:21 +00:00
|
|
|
|
ptr = strchr (cmd, '>');
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (ptr) {
|
|
|
|
|
ptr[0] = '\0';
|
2010-02-28 21:58:21 +00:00
|
|
|
|
str = r_str_trim_head_tail (ptr+1+(ptr[1]=='>'));
|
|
|
|
|
pipefd = r_cons_pipe_open (str, ptr[1]=='>');
|
|
|
|
|
ret = r_core_cmd_subst (core, cmd);
|
|
|
|
|
r_cons_flush ();
|
|
|
|
|
r_cons_pipe_close (pipefd);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
2009-03-21 22:59:35 +00:00
|
|
|
|
|
2009-12-31 00:27:03 +00:00
|
|
|
|
/* sub commands */
|
2010-02-05 11:21:37 +00:00
|
|
|
|
ptr = strchr (cmd, '`');
|
2009-12-31 00:27:03 +00:00
|
|
|
|
if (ptr) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
ptr2 = strchr (ptr+1, '`');
|
2009-12-31 00:27:03 +00:00
|
|
|
|
if (!ptr2) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
eprintf ("parse: Missing '´' in expression.\n");
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return -1;
|
|
|
|
|
} else {
|
|
|
|
|
ptr[0] = '\0';
|
|
|
|
|
ptr2[0] = '\0';
|
2010-02-05 11:21:37 +00:00
|
|
|
|
str = r_core_cmd_str (core, ptr+1);
|
|
|
|
|
for(i=0;str[i];i++)
|
|
|
|
|
if (str[i]=='\n') str[i]=' ';
|
|
|
|
|
cmd = r_str_concat (strdup (cmd), r_str_concat (str, ptr2+1));
|
|
|
|
|
ret = r_core_cmd_subst (core, cmd);
|
|
|
|
|
free (cmd);
|
|
|
|
|
free (str);
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return ret;
|
2009-03-21 22:59:35 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-12-31 00:27:03 +00:00
|
|
|
|
/* grep the content */
|
2010-02-05 11:21:37 +00:00
|
|
|
|
ptr = strchr (cmd, '~');
|
2009-03-21 22:59:35 +00:00
|
|
|
|
if (ptr) {
|
|
|
|
|
ptr[0]='\0';
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_cons_grep (ptr+1);
|
|
|
|
|
} else r_cons_grep (NULL);
|
2009-03-21 22:59:35 +00:00
|
|
|
|
|
2009-12-31 00:27:03 +00:00
|
|
|
|
/* seek commands */
|
2010-02-05 11:21:37 +00:00
|
|
|
|
ptr = strchr (cmd, '@');
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (ptr) {
|
|
|
|
|
ptr[0]='\0';
|
2010-01-12 01:12:18 +00:00
|
|
|
|
ut64 tmpoff = core->offset;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
if (ptr[1]=='@') {
|
2009-03-21 02:17:43 +00:00
|
|
|
|
// TODO: remove temporally seek (should be done by cmd_foreach)
|
2010-03-05 12:18:44 +00:00
|
|
|
|
ret = r_core_cmd_foreach (core, cmd, ptr+2);
|
|
|
|
|
//ret = -1; /* do not run out-of-foreach cmd */
|
2009-12-31 00:27:03 +00:00
|
|
|
|
} else {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (r_core_seek (core, r_num_math (core->num, ptr+1), 1))
|
|
|
|
|
ret = r_cmd_call (core->cmd, r_str_trim_head (cmd));
|
2010-04-08 12:04:34 +00:00
|
|
|
|
else ret = 0;
|
2009-12-31 00:27:03 +00:00
|
|
|
|
}
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_core_seek (core, tmpoff, 1);
|
2010-04-08 12:04:34 +00:00
|
|
|
|
ptr[0]='@';
|
2009-12-31 00:27:03 +00:00
|
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
2009-03-20 21:05:12 +00:00
|
|
|
|
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ret = r_cmd_call (core->cmd, r_str_trim_head (cmd));
|
2010-03-07 23:18:58 +00:00
|
|
|
|
//r_cons_grep_end ();
|
|
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
R_API int r_core_cmd_foreach(RCore *core, const char *cmd, char *each) {
|
2009-03-20 21:05:12 +00:00
|
|
|
|
int i=0,j;
|
|
|
|
|
char ch;
|
|
|
|
|
char *word = NULL;
|
2009-03-21 22:59:35 +00:00
|
|
|
|
char *str, *ostr;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
struct list_head *pos;
|
2009-07-08 11:49:55 +00:00
|
|
|
|
ut64 oseek, addr;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
|
2010-02-28 21:58:21 +00:00
|
|
|
|
for (; *each==' '; each++);
|
|
|
|
|
for (; *cmd==' '; cmd++);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
|
2010-01-12 01:12:18 +00:00
|
|
|
|
oseek = core->offset;
|
2009-03-21 22:59:35 +00:00
|
|
|
|
ostr = str = strdup(each);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
//radare_controlc();
|
|
|
|
|
|
2010-02-28 21:58:21 +00:00
|
|
|
|
switch (each[0]) {
|
2009-03-20 21:05:12 +00:00
|
|
|
|
case '?':
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Foreach '@@' iterator command:\n"
|
|
|
|
|
" This command is used to repeat a command over a list of offsets.\n"
|
|
|
|
|
" x @@ sym. Run 'x' over all flags matching 'sym.'\n"
|
|
|
|
|
" x @@.file \"\" over the offsets specified in the file (one offset per line)\n"
|
|
|
|
|
" x @@=off1 off2 .. Manual list of offsets\n"
|
|
|
|
|
" x @@=`pdf~call[0]` Run 'x' at every call offset of the current function\n");
|
2009-03-20 21:05:12 +00:00
|
|
|
|
break;
|
|
|
|
|
case '=':
|
|
|
|
|
/* foreach list of items */
|
2009-03-21 22:59:35 +00:00
|
|
|
|
each = str+1;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
do {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
while (each[0]==' ')
|
|
|
|
|
each = each+1;
|
2009-03-21 22:59:35 +00:00
|
|
|
|
if (!*each) break;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
str = strchr (each, ' ');
|
2009-03-21 02:17:43 +00:00
|
|
|
|
if (str) {
|
2009-03-20 21:05:12 +00:00
|
|
|
|
str[0]='\0';
|
2010-05-19 22:59:42 +00:00
|
|
|
|
addr = r_num_math (core->num, each);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
str[0]=' ';
|
2010-05-19 22:59:42 +00:00
|
|
|
|
} else addr = r_num_math (core->num, each);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("; 0x%08"PFMT64x":\n", addr);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
each = str+1;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_core_seek (core, addr, 1);
|
|
|
|
|
r_core_cmd (core, cmd, 0);
|
|
|
|
|
r_cons_flush ();
|
|
|
|
|
} while (str != NULL);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
break;
|
|
|
|
|
case '.':
|
|
|
|
|
if (each[1]=='(') {
|
|
|
|
|
char cmd2[1024];
|
|
|
|
|
// TODO: use controlc() here
|
2010-02-05 11:21:37 +00:00
|
|
|
|
// XXX whats this 999 ?
|
2010-05-19 22:59:42 +00:00
|
|
|
|
for(core->cmd->macro.counter=0;i<999;core->cmd->macro.counter++) {
|
|
|
|
|
r_cmd_macro_call (&core->cmd->macro, each+2);
|
|
|
|
|
if (core->cmd->macro.brk_value == NULL)
|
2009-03-20 21:05:12 +00:00
|
|
|
|
break;
|
|
|
|
|
|
2010-05-19 22:59:42 +00:00
|
|
|
|
addr = core->cmd->macro._brk_value;
|
2010-04-14 11:02:23 +00:00
|
|
|
|
sprintf (cmd2, "%s @ 0x%08"PFMT64x"", cmd, addr);
|
|
|
|
|
eprintf ("0x%08"PFMT64x" (%s)\n", addr, cmd2);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_core_seek (core, addr, 1);
|
|
|
|
|
r_core_cmd (core, cmd2, 0);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
i++;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
char buf[1024];
|
|
|
|
|
char cmd2[1024];
|
2010-02-05 11:21:37 +00:00
|
|
|
|
FILE *fd = fopen (each+1, "r");
|
2009-09-24 10:29:05 +00:00
|
|
|
|
if (fd) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->cmd->macro.counter=0;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
while (!feof (fd)) {
|
|
|
|
|
buf[0] = '\0';
|
|
|
|
|
if (fgets (buf, 1024, fd) == NULL)
|
|
|
|
|
break;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
addr = r_num_math (core->num, buf);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("0x%08"PFMT64x": %s\n", addr, cmd);
|
|
|
|
|
sprintf (cmd2, "%s @ 0x%08"PFMT64x"", cmd, addr);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_core_seek (core, addr, 1); // XXX
|
|
|
|
|
r_core_cmd (core, cmd2, 0);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->cmd->macro.counter++;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
}
|
2010-02-05 11:21:37 +00:00
|
|
|
|
fclose (fd);
|
|
|
|
|
} else eprintf ("Cannot open file '%s' to read offsets\n", each+1);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->cmd->macro.counter = 0;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
//while(str[i]) && !core->interrupted) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
while (str[i]) {
|
2009-03-20 21:05:12 +00:00
|
|
|
|
j = i;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
for (;str[j]&&str[j]==' ';j++); // skip spaces
|
|
|
|
|
for (i=j;str[i]&&str[i]!=' ';i++); // find EOS
|
2009-03-20 21:05:12 +00:00
|
|
|
|
ch = str[i];
|
|
|
|
|
str[i] = '\0';
|
2010-02-05 11:21:37 +00:00
|
|
|
|
word = strdup (str+j);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
if (word == NULL)
|
|
|
|
|
break;
|
|
|
|
|
str[i] = ch;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
if (strchr (word, '*')) {
|
2009-03-20 21:05:12 +00:00
|
|
|
|
#if 0
|
|
|
|
|
/* for all flags in current flagspace */
|
|
|
|
|
list_for_each(pos, &flags) {
|
|
|
|
|
flag_t *flag = (flag_t *)list_entry(pos, flag_t, list);
|
|
|
|
|
//if (core->interrupted)
|
|
|
|
|
// break;
|
|
|
|
|
/* filter per flag spaces */
|
|
|
|
|
// if ((flag_space_idx != -1) && (flag->space != flag_space_idx))
|
|
|
|
|
// continue;
|
|
|
|
|
|
2010-01-31 13:22:27 +00:00
|
|
|
|
core->offset = flag->offset;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
radare_read(0);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
cons_printf("; @@ 0x%08"PFMT64x" (%s)\n", core->offset, flag->name);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
radare_cmd(cmd,0);
|
|
|
|
|
}
|
|
|
|
|
#else
|
2010-02-22 23:26:13 +00:00
|
|
|
|
eprintf ("No flags foreach implemented\n");
|
2009-03-20 21:05:12 +00:00
|
|
|
|
#endif
|
|
|
|
|
} else {
|
|
|
|
|
/* for all flags in current flagspace */
|
2010-05-19 22:59:42 +00:00
|
|
|
|
list_for_each (pos, &core->flags->flags) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
RFlagItem *flag = (RFlagItem *)list_entry(pos, RFlagItem, list);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
|
2010-02-18 15:36:55 +00:00
|
|
|
|
if (r_cons_singleton()->breaked)
|
2009-03-21 02:17:43 +00:00
|
|
|
|
break;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
/* filter per flag spaces */
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if ((core->flags->space_idx != -1) && (flag->space != core->flags->space_idx))
|
2009-03-20 21:05:12 +00:00
|
|
|
|
continue;
|
|
|
|
|
if (word[0]=='\0' || strstr(flag->name, word) != NULL) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_core_seek (core, flag->offset, 1);
|
2010-04-08 13:48:53 +00:00
|
|
|
|
// TODO: Debug mode print
|
2010-04-14 11:02:23 +00:00
|
|
|
|
//r_cons_printf ("# @@ 0x%08"PFMT64x" (%s)\n", core->offset, flag->name);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_core_cmd (core, cmd, 0);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#if 0
|
|
|
|
|
/* ugly copypasta from tmpseek .. */
|
|
|
|
|
if (strstr(word, each)) {
|
|
|
|
|
if (word[i]=='+'||word[i]=='-')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->offset = core->offset + r_num_math (get_math(core->num, word);
|
|
|
|
|
else core->offset = r_num_math (get_math(core->num, word);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
radare_read(0);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
cons_printf("; @@ 0x%08"PFMT64x"\n", core->offset);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
radare_cmd(cmd,0);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_cons_break (NULL, NULL);
|
2009-03-20 21:05:12 +00:00
|
|
|
|
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->cmd->macro.counter++ ;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
free (word);
|
|
|
|
|
word = NULL;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-02-05 11:21:37 +00:00
|
|
|
|
r_cons_break_end ();
|
2009-03-21 02:17:43 +00:00
|
|
|
|
// XXX: use r_core_seek here
|
2010-01-12 01:12:18 +00:00
|
|
|
|
core->offset = oseek;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
|
2010-02-05 11:21:37 +00:00
|
|
|
|
free (word);
|
|
|
|
|
free (ostr);
|
2009-04-02 10:23:32 +00:00
|
|
|
|
return R_TRUE;
|
2009-03-20 21:05:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-03-19 03:02:23 +00:00
|
|
|
|
R_API int r_core_cmd(RCore *core, const char *command, int log) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
int len, rep, ret = R_FALSE;
|
2010-01-21 01:38:52 +00:00
|
|
|
|
char *cmd, *ocmd;
|
2010-02-05 11:21:37 +00:00
|
|
|
|
if (command != NULL) {
|
|
|
|
|
len = strlen (command)+1;
|
|
|
|
|
ocmd = cmd = malloc (len+8192);
|
|
|
|
|
if (ocmd == NULL)
|
|
|
|
|
return R_FALSE;
|
|
|
|
|
memcpy (cmd, command, len);
|
|
|
|
|
cmd = r_str_trim_head_tail (cmd);
|
|
|
|
|
|
|
|
|
|
rep = atoi (cmd);
|
|
|
|
|
if (rep<1) rep = 1;
|
|
|
|
|
if (rep>0) {
|
|
|
|
|
ret = R_TRUE;
|
|
|
|
|
while (*cmd>='0'&&*cmd<='9')
|
|
|
|
|
cmd++;
|
|
|
|
|
while (rep--) {
|
|
|
|
|
ret = r_core_cmd_subst (core, cmd);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (ret<0)
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
2010-01-21 01:38:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-05 11:21:37 +00:00
|
|
|
|
if (log) r_line_hist_add (command);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
2010-02-05 11:21:37 +00:00
|
|
|
|
free (core->oobi);
|
|
|
|
|
free (ocmd);
|
|
|
|
|
core->oobi = NULL;
|
|
|
|
|
core->oobi_len = 0;
|
|
|
|
|
}
|
2009-09-24 10:29:05 +00:00
|
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 21:12:19 +00:00
|
|
|
|
R_API int r_core_cmd_file(RCore *core, const char *file) {
|
2010-03-30 22:03:59 +00:00
|
|
|
|
int ret = R_TRUE;
|
|
|
|
|
char *data, *odata = r_file_slurp (file, NULL);
|
|
|
|
|
if (odata != NULL) {
|
|
|
|
|
char *nl = strchr (odata, '\n');
|
|
|
|
|
if (nl) {
|
|
|
|
|
data = odata;
|
|
|
|
|
do {
|
|
|
|
|
*nl = '\0';
|
|
|
|
|
if (r_core_cmd (core, data, 0) == -1) {
|
|
|
|
|
eprintf ("r_core_cmd_file: Failed to run '%s'\n", data);
|
|
|
|
|
ret = R_FALSE;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
data = nl+1;
|
|
|
|
|
} while ((nl = strchr (data, '\n')));
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
2010-03-30 22:03:59 +00:00
|
|
|
|
free (odata);
|
|
|
|
|
} else ret = R_FALSE;
|
|
|
|
|
return ret;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 21:12:19 +00:00
|
|
|
|
R_API int r_core_cmd_command(RCore *core, const char *command) {
|
2009-12-30 10:03:18 +00:00
|
|
|
|
int len;
|
2010-02-15 21:59:26 +00:00
|
|
|
|
char *buf, *rcmd, *ptr;
|
|
|
|
|
rcmd = ptr = buf = r_sys_cmd_str (command, 0, &len);
|
2009-12-30 10:03:18 +00:00
|
|
|
|
if (buf == NULL)
|
|
|
|
|
return -1;
|
2010-01-21 01:38:52 +00:00
|
|
|
|
while ((ptr = strstr (rcmd, "\n"))) {
|
2009-12-30 10:03:18 +00:00
|
|
|
|
*ptr = '\0';
|
2010-01-21 01:38:52 +00:00
|
|
|
|
if (r_core_cmd (core, rcmd, 0) == -1) {
|
|
|
|
|
eprintf ("Error running command '%s'\n", rcmd);
|
2010-02-15 21:59:26 +00:00
|
|
|
|
break;
|
2009-12-30 10:03:18 +00:00
|
|
|
|
}
|
2010-01-21 01:38:52 +00:00
|
|
|
|
rcmd += strlen (rcmd)+1;
|
2009-12-30 10:03:18 +00:00
|
|
|
|
}
|
|
|
|
|
r_str_free(buf);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-05 11:21:37 +00:00
|
|
|
|
static void cmd_dm(RCore *core, const char *input) {
|
|
|
|
|
switch (input[0]) {
|
|
|
|
|
case '?':
|
|
|
|
|
r_cons_printf (
|
|
|
|
|
"Usage: dm [size]\n"
|
|
|
|
|
" dm List memory maps of target process\n"
|
|
|
|
|
" dm* Same as above but in radare commands\n"
|
|
|
|
|
" dm 4096 Allocate 4096 bytes in child process\n"
|
|
|
|
|
" dm-0x8048 Deallocate memory map of address 0x8048\n"
|
2010-03-05 00:49:12 +00:00
|
|
|
|
//" dm rw- esp 9K set 9KB of the stack as read+write (no exec)\n"
|
2010-02-05 11:21:37 +00:00
|
|
|
|
"TODO: map files in process memory.\n");
|
|
|
|
|
break;
|
|
|
|
|
case '*':
|
|
|
|
|
case '-':
|
|
|
|
|
case ' ':
|
|
|
|
|
eprintf ("TODO\n");
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_map_sync (core->dbg); // update process memory maps
|
|
|
|
|
r_debug_map_list (core->dbg, core->offset);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-23 11:30:04 +00:00
|
|
|
|
static int step_until(RCore *core, ut64 addr) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
ut64 off = r_debug_reg_get (core->dbg, "pc");
|
2010-03-23 11:30:04 +00:00
|
|
|
|
if (off == 0LL) {
|
|
|
|
|
eprintf ("Cannot 'drn pc'\n");
|
|
|
|
|
return R_FALSE;
|
|
|
|
|
}
|
|
|
|
|
if (addr == 0LL) {
|
|
|
|
|
eprintf ("Cannot continue until address 0\n");
|
|
|
|
|
return R_FALSE;
|
|
|
|
|
}
|
|
|
|
|
do {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_step (core->dbg, 1);
|
|
|
|
|
off = r_debug_reg_get (core->dbg, "pc");
|
2010-03-23 11:30:04 +00:00
|
|
|
|
// check breakpoint here
|
|
|
|
|
} while (off != addr);
|
|
|
|
|
return R_TRUE;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-28 21:58:21 +00:00
|
|
|
|
static int step_line(RCore *core, int times) {
|
|
|
|
|
char file[512], file2[512];
|
|
|
|
|
int find_meta, line, line2;
|
|
|
|
|
ut64 off;
|
|
|
|
|
|
2010-05-19 22:59:42 +00:00
|
|
|
|
off = r_debug_reg_get (core->dbg, "pc");
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (off == 0LL) {
|
|
|
|
|
eprintf ("Cannot 'drn pc'\n");
|
|
|
|
|
return R_FALSE;
|
|
|
|
|
}
|
2010-05-19 22:59:42 +00:00
|
|
|
|
if (r_bin_meta_get_line (core->bin, off, file, sizeof (file), &line)) {
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("--> 0x%08"PFMT64x" %s : %d\n", off, file, line);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("--> %s\n", r_file_slurp_line (file, line, 0));
|
|
|
|
|
find_meta = R_FALSE;
|
|
|
|
|
} else {
|
|
|
|
|
eprintf ("--> Stepping until dwarf line\n");
|
|
|
|
|
find_meta = R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
do {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_step (core->dbg, 1);
|
|
|
|
|
off = r_debug_reg_get (core->dbg, "pc");
|
|
|
|
|
if (!r_bin_meta_get_line (core->bin, off, file2, sizeof (file2), &line2)) {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
if (find_meta)
|
|
|
|
|
continue;
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("Cannot retrieve dwarf info at 0x%08"PFMT64x"\n", off);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
return R_FALSE;
|
|
|
|
|
}
|
|
|
|
|
} while (!strcmp (file, file2) && line == line2);
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("--> 0x%08"PFMT64x" %s : %d\n", off, file2, line2);
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("--> %s\n", r_file_slurp_line (file2, line2, 0));
|
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-12 17:46:11 +00:00
|
|
|
|
static void cmd_debug_pid(RCore *core, const char *input) {
|
|
|
|
|
const char *ptr;
|
|
|
|
|
int pid, sig;
|
|
|
|
|
if (input[1] == 'k') {
|
|
|
|
|
/* XXX: not for threads? signal is for a whole process!! */
|
|
|
|
|
/* XXX: but we want fine-grained access to process resources */
|
|
|
|
|
pid = atoi (input+2);
|
|
|
|
|
ptr = strchr (input, ' ');
|
|
|
|
|
if (ptr) sig = atoi (ptr+1);
|
|
|
|
|
else sig = 0;
|
|
|
|
|
if (pid > 0) {
|
|
|
|
|
eprintf ("Sending signal '%d' to pid '%d'\n",
|
|
|
|
|
sig, pid);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_kill (core->dbg, sig);
|
2010-03-12 17:46:11 +00:00
|
|
|
|
} else eprintf ("Invalid arguments\n");
|
|
|
|
|
} else
|
|
|
|
|
if (input[1] == 't') {
|
|
|
|
|
if (input[2]=='=' || input[2]==' ')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_select (core->dbg,
|
|
|
|
|
(int) r_num_math (core->num, input+3),
|
|
|
|
|
(int) r_num_math (core->num, input+3));
|
|
|
|
|
else r_debug_thread_list (core->dbg, core->dbg->pid);
|
2010-03-12 17:46:11 +00:00
|
|
|
|
} else
|
|
|
|
|
if (input[1]=='?')
|
|
|
|
|
r_cons_printf ("Usage: dp[=][pid]\n"
|
|
|
|
|
" dp list current pid and childrens\n"
|
|
|
|
|
" dp 748 list childs of pid\n"
|
|
|
|
|
" dp* list all attachable pids\n"
|
|
|
|
|
" dpa 377 attach and select this pid\n"
|
|
|
|
|
" dp=748 select this pid\n"
|
|
|
|
|
" dpt List threads of current pid\n"
|
|
|
|
|
" dpt 74 List threads of given process\n"
|
|
|
|
|
" dpt=64 Attach to thread\n"
|
|
|
|
|
" dpk P S send signal S to P process id\n");
|
|
|
|
|
else
|
|
|
|
|
if (input[1]=='a') {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_attach (core->dbg,
|
|
|
|
|
(int) r_num_math (core->num, input+2));
|
|
|
|
|
r_debug_select (core->dbg,
|
|
|
|
|
(int) r_num_math (core->num, input+2),
|
|
|
|
|
(int) r_num_math (core->num, input+2));
|
2010-03-12 17:46:11 +00:00
|
|
|
|
} else
|
2010-04-10 23:46:07 +00:00
|
|
|
|
if (input[1]=='f')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_select (core->dbg, core->file->fd, core->file->fd);
|
2010-04-10 23:46:07 +00:00
|
|
|
|
else
|
2010-03-12 17:46:11 +00:00
|
|
|
|
if (input[1]=='=')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_select (core->dbg,
|
|
|
|
|
(int) r_num_math (core->num, input+2),
|
|
|
|
|
(int) r_num_math (core->num, input+2));
|
2010-03-12 17:46:11 +00:00
|
|
|
|
else
|
|
|
|
|
if (input[1]=='*')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_pid_list (core->dbg, 0);
|
2010-03-12 17:46:11 +00:00
|
|
|
|
else
|
|
|
|
|
if (input[1]==' ')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_pid_list (core->dbg,
|
|
|
|
|
(int) r_num_math (core->num, input+2));
|
|
|
|
|
else r_debug_pid_list (core->dbg, core->dbg->pid);
|
2010-03-12 17:46:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-09-25 02:04:51 +00:00
|
|
|
|
static int cmd_debug(void *data, const char *input) {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
RCore *core = (RCore *)data;
|
2010-03-03 10:35:03 +00:00
|
|
|
|
struct r_anal_aop_t analop;
|
2010-03-12 17:46:11 +00:00
|
|
|
|
int len, times, sig;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
ut64 addr;
|
2009-04-01 22:44:43 +00:00
|
|
|
|
char *ptr;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
|
2010-01-19 10:25:17 +00:00
|
|
|
|
switch (input[0]) {
|
2010-02-05 11:21:37 +00:00
|
|
|
|
case 'x':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_execute (core->dbg, (ut8*)
|
2010-02-15 21:59:26 +00:00
|
|
|
|
"\xc7\xc0\x03\x00\x00\x00\x33\xdb\x33"
|
|
|
|
|
"\xcc\xc7\xc2\x10\x00\x00\x00\xcd\x80", 18);
|
2010-02-05 11:21:37 +00:00
|
|
|
|
break;
|
2010-03-12 17:46:11 +00:00
|
|
|
|
case 't':
|
|
|
|
|
// TODO: Add support to change the tag
|
|
|
|
|
if (input[1]=='r')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_trace_reset (core->dbg, R_TRUE);
|
|
|
|
|
else r_debug_trace_list (core->dbg, -1);
|
2010-03-01 09:49:04 +00:00
|
|
|
|
break;
|
2010-03-07 23:18:58 +00:00
|
|
|
|
case 'd':
|
|
|
|
|
eprintf ("TODO: dd: file descriptors\n");
|
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 's':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
times = atoi (input+2);
|
|
|
|
|
if (times<1) times = 1;
|
2010-03-07 23:18:58 +00:00
|
|
|
|
if (input[1]=='?')
|
|
|
|
|
r_cons_printf ("Usage: ds[ol] [count]\n"
|
2010-03-23 11:30:04 +00:00
|
|
|
|
" ds step one instruction\n"
|
|
|
|
|
" ds 4 step 4 instructions\n"
|
|
|
|
|
" dso 3 step over 3 instructions\n"
|
|
|
|
|
" dsu addr step until address\n"
|
|
|
|
|
" dsl step one source line\n"
|
|
|
|
|
" dsl 40 step 40 source lines\n");
|
|
|
|
|
else if (input[1]=='u') {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
step_until (core, r_num_math (core->num, input+2)); // XXX dupped by times
|
2010-03-23 11:30:04 +00:00
|
|
|
|
} else if (input[1]=='o')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_step_over (core->dbg, times);
|
2010-03-23 11:30:04 +00:00
|
|
|
|
else if (input[1]=='l')
|
|
|
|
|
step_line (core, times);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
else r_debug_step (core->dbg, times);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'b':
|
2010-01-19 10:25:17 +00:00
|
|
|
|
r_core_cmd_bp (core, input);
|
2009-04-12 22:46:44 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'H':
|
2010-02-28 21:58:21 +00:00
|
|
|
|
eprintf ("TODO: transplant process\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'c':
|
2010-01-19 10:25:17 +00:00
|
|
|
|
switch (input[1]) {
|
|
|
|
|
case '?':
|
|
|
|
|
eprintf("Usage: dc[?] -- continue execution\n"
|
2010-03-03 10:35:03 +00:00
|
|
|
|
" dc? show this help\n"
|
|
|
|
|
" dc continue execution of all childs\n"
|
2010-03-03 12:34:38 +00:00
|
|
|
|
" dct [len] traptrace from curseek to len, no argument to list\n"
|
2010-03-03 10:35:03 +00:00
|
|
|
|
" dcu [addr] continue until address\n"
|
2010-03-07 23:18:58 +00:00
|
|
|
|
" dco [num] step over N instructions\n"
|
2010-03-03 10:35:03 +00:00
|
|
|
|
" dcs [num] continue until syscall\n"
|
|
|
|
|
" dck [sig] [pid] continue sending kill 9 to process\n"
|
|
|
|
|
" dc [pid] continue execution of pid\n"
|
|
|
|
|
" dc[-pid] stop execution of pid\n"
|
2010-01-19 10:25:17 +00:00
|
|
|
|
"TODO: support for threads?\n");
|
|
|
|
|
break;
|
|
|
|
|
case 'k':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
// select pid and r_debug_continue_kill (core->dbg,
|
2010-01-19 10:25:17 +00:00
|
|
|
|
ptr = strchr (input+3, ' ');
|
|
|
|
|
if (ptr) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
int old_pid = core->dbg->pid;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
int pid = atoi (ptr+1);
|
|
|
|
|
*ptr = 0;
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_select (core->dbg, pid, pid);
|
|
|
|
|
r_debug_continue_kill (core->dbg, atoi (input+2));
|
|
|
|
|
r_debug_select (core->dbg, old_pid, old_pid);
|
|
|
|
|
} else r_debug_continue_kill (core->dbg, atoi (input+2));
|
2010-01-19 10:25:17 +00:00
|
|
|
|
break;
|
2010-03-02 10:18:49 +00:00
|
|
|
|
case 's':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
sig = r_num_math (core->num, input+2);
|
2010-03-02 10:18:49 +00:00
|
|
|
|
eprintf ("Continue until syscall %d\n", sig);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_continue_syscall (core->dbg, sig);
|
2010-03-02 10:18:49 +00:00
|
|
|
|
/* TODO : use r_syscall here, to retrieve syscall info */
|
|
|
|
|
break;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
case 'u':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
addr = r_num_math (core->num, input+2);
|
2010-03-19 03:02:23 +00:00
|
|
|
|
if (addr) {
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("Continue until 0x%08"PFMT64x"\n", addr);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_add_sw (core->dbg->bp, addr, 1, R_BP_PROT_EXEC);
|
|
|
|
|
r_debug_continue (core->dbg);
|
|
|
|
|
r_bp_del (core->dbg->bp, addr);
|
2010-03-19 03:02:23 +00:00
|
|
|
|
} else eprintf ("Cannot continue until address 0\n");
|
2010-02-28 21:58:21 +00:00
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
case ' ':
|
|
|
|
|
do {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
int old_pid = core->dbg->pid;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
int pid = atoi (input+2);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_select (core->dbg, pid, pid);
|
|
|
|
|
r_debug_continue (core->dbg);
|
|
|
|
|
r_debug_select (core->dbg, old_pid, old_pid);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
} while (0);
|
|
|
|
|
break;
|
2010-03-03 10:35:03 +00:00
|
|
|
|
case 't':
|
2010-05-19 22:59:42 +00:00
|
|
|
|
len = r_num_math (core->num, input+2);
|
2010-03-03 12:34:38 +00:00
|
|
|
|
if (len == 0) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_traptrace_list (core->dbg->bp);
|
2010-03-03 10:35:03 +00:00
|
|
|
|
} else {
|
2010-04-14 11:02:23 +00:00
|
|
|
|
eprintf ("Trap tracing 0x%08"PFMT64x"-0x%08"PFMT64x"\n", core->offset, core->offset+len);
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_bp_traptrace_reset (core->dbg->bp, R_TRUE);
|
|
|
|
|
r_bp_traptrace_add (core->dbg->bp, core->offset, core->offset+len);
|
|
|
|
|
r_bp_traptrace_enable (core->dbg->bp, R_TRUE);
|
2010-03-03 10:35:03 +00:00
|
|
|
|
do {
|
|
|
|
|
ut8 buf[32];
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_continue (core->dbg);
|
|
|
|
|
addr = r_debug_reg_get (core->dbg, "pc");
|
2010-03-03 10:35:03 +00:00
|
|
|
|
if (addr == 0LL) {
|
|
|
|
|
eprintf ("pc=0\n");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
/* XXX Bottleneck..we need to reuse the bytes read by traptrace */
|
|
|
|
|
// XXX Do asm.arch should define the max size of opcode?
|
|
|
|
|
r_core_read_at (core, addr, buf, 32); // XXX longer opcodes?
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_anal_aop (core->anal, &analop, addr, buf, sizeof (buf));
|
|
|
|
|
} while (r_bp_traptrace_at (core->dbg->bp, addr, analop.length));
|
|
|
|
|
r_bp_traptrace_enable (core->dbg->bp, R_FALSE);
|
2010-03-03 10:35:03 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
2010-01-19 10:25:17 +00:00
|
|
|
|
default:
|
|
|
|
|
eprintf ("continue\n");
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_continue (core->dbg);
|
2010-01-19 10:25:17 +00:00
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
2009-03-22 02:36:00 +00:00
|
|
|
|
case 'm':
|
2010-02-05 11:21:37 +00:00
|
|
|
|
cmd_dm (core, input+1);
|
2009-03-22 02:36:00 +00:00
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
case 'r':
|
2010-02-05 11:21:37 +00:00
|
|
|
|
cmd_reg (core, input+1);
|
2010-03-19 03:02:23 +00:00
|
|
|
|
//r_core_cmd (core, "|reg", 0);
|
2009-02-18 00:43:57 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'p':
|
2010-03-12 17:46:11 +00:00
|
|
|
|
cmd_debug_pid (core, input);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
case 'h':
|
2009-09-24 10:29:05 +00:00
|
|
|
|
if (input[1]==' ')
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_debug_use (core->dbg, input+2);
|
|
|
|
|
else r_debug_handle_list (core->dbg);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_cons_printf ("Usage: d[sbhcrbo] [arg]\n"
|
2010-02-03 17:15:31 +00:00
|
|
|
|
" dh [handler] list or set debugger handler\n"
|
|
|
|
|
" dH [handler] transplant process to a new handler\n"
|
2010-03-07 23:18:58 +00:00
|
|
|
|
" dd file descriptors\n"
|
|
|
|
|
" ds[ol] N step, over, source line\n"
|
2010-03-12 17:46:11 +00:00
|
|
|
|
" dp[=*?t][pid] list, attach to process or thread id\n"
|
2010-02-28 21:58:21 +00:00
|
|
|
|
" dc[?] continue execution. dc? for more\n"
|
2010-02-03 17:15:31 +00:00
|
|
|
|
" dr[?] cpu registers, dr? for extended help\n"
|
|
|
|
|
" db[?] breakpoints\n"
|
2010-03-12 17:46:11 +00:00
|
|
|
|
" dbt display backtrace\n"
|
|
|
|
|
" dt[r] [tag] display instruction traces (dtr=reset)\n"
|
2010-03-05 00:49:12 +00:00
|
|
|
|
" dm show memory maps\n");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
R_API int r_core_cmd_buffer(void *user, const char *buf) {
|
2010-02-28 21:58:21 +00:00
|
|
|
|
char *str = strdup (buf);
|
|
|
|
|
char *ptr = strchr (str, '\n');
|
2009-03-14 11:39:37 +00:00
|
|
|
|
char *optr = str;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
while (ptr) {
|
2009-03-14 11:39:37 +00:00
|
|
|
|
ptr[0]='\0';
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_core_cmd (user, optr, 0);
|
2009-03-14 11:39:37 +00:00
|
|
|
|
optr = ptr+1;
|
2010-02-28 21:58:21 +00:00
|
|
|
|
ptr = strchr (str,'\n');
|
2009-03-14 11:39:37 +00:00
|
|
|
|
}
|
2010-02-28 21:58:21 +00:00
|
|
|
|
r_core_cmd (user, optr, 0);
|
|
|
|
|
free (str);
|
2009-03-14 11:39:37 +00:00
|
|
|
|
return R_TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
R_API int r_core_cmdf(void *user, const char *fmt, ...) {
|
2009-02-18 00:43:57 +00:00
|
|
|
|
char string[1024];
|
|
|
|
|
int ret;
|
|
|
|
|
va_list ap;
|
2010-02-01 10:55:56 +00:00
|
|
|
|
va_start (ap, fmt);
|
|
|
|
|
vsnprintf (string, 1023, fmt, ap);
|
2010-03-30 21:12:19 +00:00
|
|
|
|
ret = r_core_cmd ((RCore *)user, string, 0);
|
2009-02-18 00:43:57 +00:00
|
|
|
|
va_end(ap);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-22 23:26:13 +00:00
|
|
|
|
R_API int r_core_cmd0(void *user, const char *cmd) {
|
2010-03-30 21:12:19 +00:00
|
|
|
|
return r_core_cmd ((RCore *)user, cmd, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-12-31 00:27:03 +00:00
|
|
|
|
/*
|
|
|
|
|
* return: pointer to a buffer with the output of the command.
|
|
|
|
|
*/
|
2010-03-30 21:12:19 +00:00
|
|
|
|
R_API char *r_core_cmd_str(RCore *core, const char *cmd) {
|
2009-09-20 00:16:14 +00:00
|
|
|
|
char *retstr = NULL;
|
2010-02-01 10:55:56 +00:00
|
|
|
|
r_cons_reset ();
|
|
|
|
|
if (r_core_cmd (core, cmd, 0) == -1) {
|
|
|
|
|
eprintf ("Invalid command: %s\n", cmd);
|
|
|
|
|
retstr = strdup ("");
|
2009-02-05 21:08:46 +00:00
|
|
|
|
} else {
|
|
|
|
|
const char *static_str = r_cons_get_buffer();
|
2009-09-25 02:04:51 +00:00
|
|
|
|
if (static_str==NULL)
|
2009-02-05 21:08:46 +00:00
|
|
|
|
retstr = strdup("");
|
|
|
|
|
else retstr = strdup(static_str);
|
|
|
|
|
r_cons_reset();
|
|
|
|
|
}
|
|
|
|
|
return retstr;
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 21:12:19 +00:00
|
|
|
|
int r_core_cmd_init(RCore *core) {
|
2010-05-19 22:59:42 +00:00
|
|
|
|
core->cmd = r_cmd_new ();
|
|
|
|
|
core->cmd->macro.num = core->num;
|
|
|
|
|
core->cmd->macro.user = core;
|
|
|
|
|
core->cmd->macro.cmd = r_core_cmd0;
|
|
|
|
|
r_cmd_set_data (core->cmd, core);
|
|
|
|
|
r_cmd_add (core->cmd, "x", "alias for px", &cmd_hexdump);
|
|
|
|
|
r_cmd_add (core->cmd, "analysis", "analysis", &cmd_anal);
|
|
|
|
|
r_cmd_add (core->cmd, "flag", "get/set flags", &cmd_flag);
|
|
|
|
|
r_cmd_add (core->cmd, "debug", "debugger operations", &cmd_debug);
|
|
|
|
|
r_cmd_add (core->cmd, "info", "get file info", &cmd_info);
|
|
|
|
|
r_cmd_add (core->cmd, "cmp", "compare memory", &cmd_cmp);
|
|
|
|
|
r_cmd_add (core->cmd, "seek", "seek to an offset", &cmd_seek);
|
|
|
|
|
r_cmd_add (core->cmd, "zign", "zignatures", &cmd_zign);
|
|
|
|
|
r_cmd_add (core->cmd, "Section", "setup section io information", &cmd_section);
|
|
|
|
|
r_cmd_add (core->cmd, "bsize", "change block size", &cmd_bsize);
|
|
|
|
|
r_cmd_add (core->cmd, "eval", "evaluate configuration variable", &cmd_eval);
|
|
|
|
|
r_cmd_add (core->cmd, "print", "print current block", &cmd_print);
|
|
|
|
|
r_cmd_add (core->cmd, "write", "write bytes", &cmd_write);
|
|
|
|
|
r_cmd_add (core->cmd, "Code", "code metadata", &cmd_meta);
|
|
|
|
|
r_cmd_add (core->cmd, "open", "open or map file", &cmd_open);
|
|
|
|
|
r_cmd_add (core->cmd, "yank", "yank bytes", &cmd_yank);
|
|
|
|
|
r_cmd_add (core->cmd, "Visual", "enter visual mode", &cmd_visual);
|
|
|
|
|
r_cmd_add (core->cmd, "!", "run system command", &cmd_system);
|
2010-05-19 00:39:01 +00:00
|
|
|
|
// XXX WTF DUPPED CMD!?!?! //
|
2010-05-19 22:59:42 +00:00
|
|
|
|
r_cmd_add (core->cmd, "|", "run io system command", &cmd_io_system);
|
|
|
|
|
r_cmd_add (core->cmd, "|", "io pipe", &cmd_iopipe);
|
|
|
|
|
r_cmd_add (core->cmd, "#", "calculate hash", &cmd_hash);
|
|
|
|
|
r_cmd_add (core->cmd, "?", "help message", &cmd_help);
|
|
|
|
|
r_cmd_add (core->cmd, ".", "interpret", &cmd_interpret);
|
|
|
|
|
r_cmd_add (core->cmd, "/", "search kw, pattern aes", &cmd_search);
|
|
|
|
|
r_cmd_add (core->cmd, "(", "macro", &cmd_macro);
|
|
|
|
|
r_cmd_add (core->cmd, "quit", "exit program session", &cmd_quit);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|