mirror of
https://github.com/radareorg/radare2.git
synced 2024-12-13 16:18:33 +00:00
fc6e799c26
Simplify r_print_string Fix r_io write boundaries
893 lines
26 KiB
C
893 lines
26 KiB
C
/* radare - LGPL - Copyright 2009-2013 - pancake */
|
|
|
|
static int printzoomcallback(void *user, int mode, ut64 addr, ut8 *bufz, ut64 size) {
|
|
RCore *core = (RCore *) user;
|
|
int j, ret = 0;
|
|
RListIter *iter;
|
|
RFlagItem *flag;
|
|
|
|
switch (mode) {
|
|
case 'p':
|
|
for (j=0; j<size; j++)
|
|
if (IS_PRINTABLE (bufz[j]))
|
|
ret++;
|
|
break;
|
|
case 'f':
|
|
r_list_foreach (core->flags->flags, iter, flag)
|
|
if (flag->offset <= addr && addr < flag->offset+flag->size)
|
|
ret++;
|
|
break;
|
|
case 's':
|
|
j = r_flag_space_get (core->flags, "strings");
|
|
r_list_foreach (core->flags->flags, iter, flag) {
|
|
if (flag->space == j && ((addr <= flag->offset
|
|
&& flag->offset < addr+size)
|
|
|| (addr <= flag->offset+flag->size
|
|
&& flag->offset+flag->size < addr+size)))
|
|
ret++;
|
|
}
|
|
break;
|
|
case '0': // 0xFF
|
|
for (j=0; j<size; j++)
|
|
if (bufz[j] == 0)
|
|
ret++;
|
|
break;
|
|
case 'F': // 0xFF
|
|
for (j=0; j<size; j++)
|
|
if (bufz[j] == 0xff)
|
|
ret++;
|
|
break;
|
|
case 'e': // entropy
|
|
ret = (ut8) (r_hash_entropy_fraction (bufz, size)*255);
|
|
break;
|
|
case 'h': // head
|
|
default:
|
|
ret = *bufz;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
R_API void r_core_print_cmp(RCore *core, ut64 from, ut64 to) {
|
|
long int delta = 0;
|
|
int col = core->cons->columns>123;
|
|
ut8 *b = malloc (core->blocksize);
|
|
ut64 addr = core->offset;
|
|
memset (b, 0xff, core->blocksize);
|
|
delta = addr - from;
|
|
r_core_read_at (core, to+delta, b, core->blocksize);
|
|
r_print_hexdiff (core->print, core->offset, core->block, to+delta, b, core->blocksize, col);
|
|
free (b);
|
|
}
|
|
|
|
static int cmd_print(void *data, const char *input) {
|
|
RCore *core = (RCore *)data;
|
|
int mode, w, p, i, l, len, total[10];
|
|
ut64 off, from, to, at, ate, piece;
|
|
ut32 tbs = core->blocksize;
|
|
ut8 *ptr = core->block;
|
|
RCoreAnalStats *as;
|
|
ut64 n;
|
|
|
|
/* TODO: Change also blocksize for 'pd'.. */
|
|
l = len = core->blocksize;
|
|
if (input[0] && input[1]) {
|
|
const char *p = strchr (input, ' ');
|
|
if (p) {
|
|
l = (int) r_num_math (core->num, p+1);
|
|
/* except disasm and memoryfmt (pd, pm) */
|
|
if (input[0] != 'd' && input[0] != 'm') {
|
|
if (l>0) len = l;
|
|
if (l>tbs) r_core_block_size (core, l);
|
|
l = len;
|
|
}
|
|
}// else l = 0;
|
|
} else l = len;
|
|
|
|
n = r_config_get_i (core->config, "io.maxblk");
|
|
i = (int)n;
|
|
if (i != n) i = 0;
|
|
if (i && l > i) {
|
|
eprintf ("This block size is too big (%d<%d). Did you mean 'p%c @ %s' instead?\n",
|
|
i, l, *input, input+2);
|
|
return R_FALSE;
|
|
}
|
|
|
|
if (input[0] && input[0]!='Z' && input[1] == 'f') {
|
|
RAnalFunction *f = r_anal_fcn_find (core->anal, core->offset,
|
|
R_ANAL_FCN_TYPE_FCN|R_ANAL_FCN_TYPE_SYM);
|
|
if (f) len = f->size;
|
|
else eprintf ("Cannot find function at 0x%08"PFMT64x"\n", core->offset);
|
|
}
|
|
ptr = core->block;
|
|
core->num->value = len;
|
|
switch (*input) {
|
|
case 'v':
|
|
mode = input[1];
|
|
w = core->print->cols * 4;
|
|
if (mode == 'j')
|
|
r_cons_strcat ("{");
|
|
off = core->offset;
|
|
for (i=0; i<10; i++) total[i] = 0;
|
|
|
|
r_core_get_boundaries (core, "file", &from, &to);
|
|
piece = (to-from) / w;
|
|
if (piece<1) piece = 1;
|
|
as = r_core_anal_get_stats (core, from, to, piece);
|
|
//eprintf ("RANGE = %llx %llx\n", from, to);
|
|
switch (mode) {
|
|
case '?':
|
|
r_cons_printf ("Usage: p%%[jh] [pieces]\n");
|
|
r_cons_printf (" pv show ascii-art bar of metadata in file boundaries\n");
|
|
r_cons_printf (" pvj show json format\n");
|
|
r_cons_printf (" pvh show histogram analysis of metadata per block\n");
|
|
return 0;
|
|
case 'j':
|
|
r_cons_printf (
|
|
"\"from\":%"PFMT64d","
|
|
"\"to\":%"PFMT64d","
|
|
"\"blocksize\":%d,"
|
|
"\"blocks\":[", from, to, piece);
|
|
break;
|
|
case 'h':
|
|
r_cons_printf (".-------------.---------------------------------.\n");
|
|
r_cons_printf ("| offset | flags funcs cmts imps syms str |\n");
|
|
r_cons_printf ("|-------------)---------------------------------|\n");
|
|
//r_cons_printf ("__offset___|_flags_funcs_cmts_imps_syms_str__|\n");
|
|
break;
|
|
default:
|
|
r_cons_printf ("0x%"PFMT64x" [", from);
|
|
}
|
|
|
|
len = 0;
|
|
for (i=0; i<w; i++) {
|
|
at = from + (piece*i);
|
|
ate = at + piece;
|
|
p = (at-from)/piece;
|
|
switch (mode) {
|
|
case 'j':
|
|
r_cons_printf ("%s{",len?",":"");
|
|
if ((as->block[p].flags)
|
|
|| (as->block[p].functions)
|
|
|| (as->block[p].comments)
|
|
|| (as->block[p].imports)
|
|
|| (as->block[p].symbols)
|
|
|| (as->block[p].strings))
|
|
r_cons_printf ("\"offset\":%"PFMT64d",", at), l++;
|
|
// TODO: simplify with macro
|
|
l = 0;
|
|
if (as->block[p].flags) r_cons_printf ("%s\"flags\":%d", l?",":"", as->block[p].flags), l++;
|
|
if (as->block[p].functions) r_cons_printf ("%s\"functions\":%d", l?",":"", as->block[p].functions), l++;
|
|
if (as->block[p].comments) r_cons_printf ("%s\"comments\":%d", l?",":"", as->block[p].comments), l++;
|
|
if (as->block[p].imports) r_cons_printf ("%s\"imports\":%d", l?",":"", as->block[p].imports), l++;
|
|
if (as->block[p].symbols) r_cons_printf ("%s\"symbols\":%d", l?",":"", as->block[p].symbols), l++;
|
|
if (as->block[p].strings) r_cons_printf ("%s\"strings\":%d", l?",":"", as->block[p].strings), l++;
|
|
r_cons_strcat ("}");
|
|
len++;
|
|
break;
|
|
case 'h':
|
|
total[0] += as->block[p].flags;
|
|
total[1] += as->block[p].functions;
|
|
total[2] += as->block[p].comments;
|
|
total[3] += as->block[p].imports;
|
|
total[4] += as->block[p].symbols;
|
|
total[5] += as->block[p].strings;
|
|
if ((as->block[p].flags)
|
|
|| (as->block[p].functions)
|
|
|| (as->block[p].comments)
|
|
|| (as->block[p].imports)
|
|
|| (as->block[p].symbols)
|
|
|| (as->block[p].strings))
|
|
r_cons_printf ("| 0x%09"PFMT64x" | %4d %4d %4d %4d %4d %4d |\n", at,
|
|
as->block[p].flags,
|
|
as->block[p].functions,
|
|
as->block[p].comments,
|
|
as->block[p].imports,
|
|
as->block[p].symbols,
|
|
as->block[p].strings);
|
|
break;
|
|
default:
|
|
if (off>=at && off<ate) {
|
|
r_cons_memcat ("^", 1);
|
|
} else {
|
|
if (as->block[p].strings>0)
|
|
r_cons_memcat ("z", 1);
|
|
else if (as->block[p].imports>0)
|
|
r_cons_memcat ("i", 1);
|
|
else if (as->block[p].symbols>0)
|
|
r_cons_memcat ("s", 1);
|
|
else if (as->block[p].functions>0)
|
|
r_cons_memcat ("F", 1);
|
|
else if (as->block[p].flags>0)
|
|
r_cons_memcat ("f", 1);
|
|
else if (as->block[p].comments>0)
|
|
r_cons_memcat ("c", 1);
|
|
else r_cons_memcat (".", 1);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
switch (mode) {
|
|
case 'j':
|
|
r_cons_strcat ("]}\n");
|
|
break;
|
|
case 'h':
|
|
//r_cons_printf (" total | flags funcs cmts imps syms str |\n");
|
|
r_cons_printf ("|-------------)---------------------------------|\n");
|
|
r_cons_printf ("| total | %4d %4d %4d %4d %4d %4d |\n",
|
|
total[0], total[1], total[2], total[3], total[4], total[5]);
|
|
r_cons_printf ("`-------------'---------------------------------'\n");
|
|
break;
|
|
default:
|
|
r_cons_printf ("] 0x%"PFMT64x"\n", to);
|
|
}
|
|
r_core_anal_stats_free (as);
|
|
break;
|
|
case '=':
|
|
switch (input[1]) {
|
|
case '?': // entropy
|
|
eprintf ("Usage: p=[bep?]\n");
|
|
eprintf (" p= print bytes of current block in bars\n");
|
|
eprintf (" p=b same as above\n");
|
|
eprintf (" p=e same as above but with entropy\n");
|
|
eprintf (" p=p print number of printable bytes for each filesize/blocksize\n");
|
|
break;
|
|
case 'e': // entropy
|
|
{
|
|
ut8 *p;
|
|
int psz, i = 0;
|
|
int fsz = core->file?core->file->size:0;
|
|
psz = fsz / core->blocksize;
|
|
ptr = malloc (core->blocksize);
|
|
eprintf ("offset = num * %d\n", psz);
|
|
p = malloc (psz);
|
|
for (i=0; i<core->blocksize; i++) {
|
|
r_core_read_at (core, i*psz, p, psz);
|
|
ptr[i] = (ut8) (256 * r_hash_entropy_fraction (p, psz));
|
|
}
|
|
free (p);
|
|
}
|
|
break;
|
|
case 'p': // printable chars
|
|
{
|
|
ut8 *p;
|
|
int psz, i = 0, j, k;
|
|
int fsz = core->file?core->file->size:0;
|
|
psz = fsz/core->blocksize;
|
|
ptr = malloc (core->blocksize);
|
|
p = malloc (psz);
|
|
for (i=0; i<core->blocksize; i++) {
|
|
r_core_read_at (core, i*psz, p, psz);
|
|
for (j=k=0; j<psz; j++) {
|
|
if (IS_PRINTABLE (p[j]))
|
|
k++;
|
|
}
|
|
ptr[i] = k;
|
|
}
|
|
free (p);
|
|
}
|
|
break;
|
|
}
|
|
r_print_fill (core->print, ptr, core->blocksize);
|
|
if (ptr != core->block) {
|
|
free (ptr);
|
|
#if 0
|
|
int bsize = 512;
|
|
/* TODO: Reimplement using API */
|
|
char *out = r_sys_cmd_strf ("rahash2 -a entropy -b %d '%s'", bsize, core->file->filename);
|
|
if (out) {
|
|
r_cons_strcat (out);
|
|
free (out);
|
|
}
|
|
#endif
|
|
}
|
|
break;
|
|
case 'a':
|
|
if (input[1]=='d') {
|
|
RAsmCode *c;
|
|
r_asm_set_pc (core->assembler, core->offset);
|
|
c = r_asm_mdisassemble_hexstr (core->assembler, input+2);
|
|
if (c) {
|
|
r_cons_puts (c->buf_asm);
|
|
r_asm_code_free (c);
|
|
} else eprintf ("Invalid hexstr\n");
|
|
} else {
|
|
RAsmCode *acode;
|
|
r_asm_set_pc (core->assembler, core->offset);
|
|
acode = r_asm_massemble (core->assembler, input+1);
|
|
if (acode && *acode->buf_hex) {
|
|
r_cons_printf ("%s\n", acode->buf_hex);
|
|
r_asm_code_free (acode);
|
|
}
|
|
}
|
|
break;
|
|
case 'b': {
|
|
const int size = len*8;
|
|
char *buf = malloc (size+1);
|
|
if (buf) {
|
|
r_str_bits (buf, core->block, size, NULL);
|
|
r_cons_printf ("%s\n", buf);
|
|
free (buf);
|
|
} else eprintf ("ERROR: Cannot malloc %d bytes\n", size);
|
|
}
|
|
break;
|
|
case 'I':
|
|
r_core_print_disasm_instructions (core, len, l);
|
|
break;
|
|
case 'i':
|
|
r_core_print_disasm_instructions (core, core->blocksize, l);
|
|
break;
|
|
case 'D':
|
|
case 'd':
|
|
switch (input[1]) {
|
|
case 'i': {// TODO
|
|
int show_offset = r_config_get_i (core->config, "asm.offset");
|
|
int show_bytes = r_config_get_i (core->config, "asm.bytes");
|
|
int decode = r_config_get_i (core->config, "asm.decode");
|
|
const ut8 *buf = core->block;
|
|
int j, ret, err = 0;
|
|
RAsmOp asmop;
|
|
if (l==0) l = len;
|
|
for (i=j=0; j<len && j<l; i+=ret,j++) {
|
|
r_asm_set_pc (core->assembler, core->offset+i);
|
|
ret = r_asm_disassemble (core->assembler, &asmop, buf+i, core->blocksize-i);
|
|
if (show_offset)
|
|
r_cons_printf ("0x%08"PFMT64x" ", core->offset+i);
|
|
if (ret<1) {
|
|
ret = err = 1;
|
|
if (show_bytes)
|
|
r_cons_printf ("%14s%02x ", "", buf[i]);
|
|
r_cons_printf ("%s\n", "???");
|
|
} else {
|
|
if (show_bytes)
|
|
r_cons_printf ("%16s ", asmop.buf_hex);
|
|
if (decode) {
|
|
RAnalOp analop = {0};
|
|
char *tmpopstr, *opstr;
|
|
r_anal_op (core->anal, &analop, core->offset+i, buf+i, core->blocksize-i);
|
|
tmpopstr = r_anal_op_to_string (core->anal, &analop);
|
|
opstr = (tmpopstr)? tmpopstr: strdup (asmop.buf_asm);
|
|
r_cons_printf ("%s\n", opstr);
|
|
free (opstr);
|
|
} else r_cons_printf ("%s\n", asmop.buf_asm);
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
break;
|
|
case 'a':
|
|
{
|
|
RAsmOp asmop;
|
|
int j, ret, err = 0;
|
|
const ut8 *buf = core->block;
|
|
if (l==0) l = len;
|
|
for (i=j=0; i<core->blocksize && j<len && j<l; i++,j++ ) {
|
|
ret = r_asm_disassemble (core->assembler, &asmop, buf+i, core->blocksize-i);
|
|
if (ret<1) {
|
|
ret = err = 1;
|
|
r_cons_printf ("???\n");
|
|
} else r_cons_printf ("0x%08"PFMT64x" %16s %s\n",
|
|
core->offset+i, asmop.buf_hex, asmop.buf_asm);
|
|
}
|
|
return R_TRUE;
|
|
}
|
|
break;
|
|
case 'r':
|
|
{
|
|
RAnalFunction *f = r_anal_fcn_find (core->anal, core->offset,
|
|
R_ANAL_FCN_TYPE_FCN|R_ANAL_FCN_TYPE_SYM);
|
|
if (f) {
|
|
RListIter *iter;
|
|
RAnalBlock *b;
|
|
// XXX: hack must be reviewed/fixed in code analysis
|
|
if (r_list_length (f->bbs) == 1) {
|
|
b = r_list_get_top (f->bbs);
|
|
if (b->size > f->size) b->size = f->size;
|
|
}
|
|
// TODO: sort by addr
|
|
r_list_foreach (f->bbs, iter, b) {
|
|
r_core_cmdf (core, "pD %"PFMT64d" @0x%"PFMT64x, b->size, b->addr);
|
|
if (b->jump != UT64_MAX)
|
|
r_cons_printf ("--> 0x%08"PFMT64x"\n", b->jump);
|
|
if (b->fail != UT64_MAX)
|
|
r_cons_printf ("--> 0x%08"PFMT64x"\n", b->fail);
|
|
r_cons_printf ("--\n");
|
|
//eprintf ( "pD %"PFMT64d" @0x%"PFMT64x"\n", b->size, b->addr);
|
|
}
|
|
} else eprintf ("Cannot find function at 0x%08"PFMT64x"\n", core->offset);
|
|
return R_TRUE;
|
|
}
|
|
break;
|
|
case 'b': {
|
|
RAnalBlock *b = r_anal_bb_from_offset (core->anal, core->offset);
|
|
if (b) {
|
|
ut8 *block = malloc (b->size+1);
|
|
if (block) {
|
|
r_core_read_at (core, b->addr, block, b->size);
|
|
core->num->value = r_core_print_disasm (
|
|
core->print, core, b->addr, block,
|
|
b->size, 9999, 0, 2);
|
|
free (block);
|
|
return 0;
|
|
}
|
|
} else eprintf ("Cannot find function at 0x%08"PFMT64x"\n", core->offset);
|
|
} break;
|
|
break;
|
|
case 'f': {
|
|
RAnalFunction *f = r_anal_fcn_find (core->anal, core->offset,
|
|
R_ANAL_FCN_TYPE_FCN|R_ANAL_FCN_TYPE_SYM);
|
|
if (f) {
|
|
ut8 *block = malloc (f->size+1);
|
|
if (block) {
|
|
r_core_read_at (core, f->addr, block, f->size);
|
|
core->num->value = r_core_print_disasm (
|
|
core->print, core, f->addr, block,
|
|
f->size, 9999, 0, 2);
|
|
free (block);
|
|
return 0;
|
|
}
|
|
} else eprintf ("Cannot find function at 0x%08"PFMT64x"\n", core->offset);
|
|
} break;
|
|
case 'l':
|
|
{
|
|
RAsmOp asmop;
|
|
int j, ret;
|
|
const ut8 *buf = core->block;
|
|
if (l==0) l= len;
|
|
for (i=j=0; i<core->blocksize && j<l; i+=ret,j++ ) {
|
|
ret = r_asm_disassemble (core->assembler, &asmop, buf+i, len-i);
|
|
printf ("%d\n", ret);
|
|
if (ret<1) ret = 1;
|
|
}
|
|
return 0;
|
|
}
|
|
break;
|
|
case 'j':
|
|
r_core_print_disasm_json (core, core->offset,
|
|
core->block, core->blocksize);
|
|
return 0;
|
|
case '?':
|
|
eprintf ("Usage: pd[f|i|l] [len] @ [addr]\n"
|
|
//TODO: eprintf (" pdr : disassemble resume\n");
|
|
" pda : disassemble all possible opcodes (byte per byte)\n"
|
|
" pdj : disassemble to json\n"
|
|
" pdb : disassemble basic block\n"
|
|
" pdr : recursive disassemble across the function graph\n"
|
|
" pdf : disassemble function\n"
|
|
" pdi : like 'pi', with offset and bytes\n"
|
|
" pdl : show instruction sizes\n");
|
|
return 0;
|
|
break;
|
|
}
|
|
//if (core->visual)
|
|
// l = core->cons->rows-core->cons->lines;
|
|
if (l<0) {
|
|
RList *bwdhits;
|
|
RListIter *iter;
|
|
RCoreAsmHit *hit;
|
|
ut8 *block = malloc (core->blocksize);
|
|
if (block) {
|
|
l = -l;
|
|
bwdhits = r_core_asm_bwdisassemble (core, core->offset, l, core->blocksize);
|
|
if (bwdhits) {
|
|
r_list_foreach (bwdhits, iter, hit) {
|
|
r_core_read_at (core, hit->addr, block, core->blocksize);
|
|
core->num->value = r_core_print_disasm (core->print,
|
|
core, hit->addr, block, core->blocksize, l, 0, 1);
|
|
r_cons_printf ("------\n");
|
|
}
|
|
r_list_free (bwdhits);
|
|
}
|
|
free (block);
|
|
}
|
|
} else {
|
|
core->num->value = r_core_print_disasm (
|
|
core->print, core, core->offset,
|
|
core->block, len, l, 0, (*input=='D'));
|
|
}
|
|
break;
|
|
case 's':
|
|
switch (input[1]) {
|
|
case '?':
|
|
r_cons_printf ("Usage: ps[zpw] [N]\n"
|
|
" ps = print string\n"
|
|
" psz = print zero terminated string\n"
|
|
" psp = print pascal string\n"
|
|
" psw = print wide string\n");
|
|
break;
|
|
case 'z':
|
|
{
|
|
char *s = malloc (core->blocksize+1);
|
|
int i, j;
|
|
if (s) {
|
|
memset (s, 0, core->blocksize);
|
|
// TODO: filter more chars?
|
|
for (i=j=0;i<core->blocksize; i++) {
|
|
char ch = (char)core->block[i];
|
|
if (!ch) break;
|
|
if (IS_PRINTABLE (ch))
|
|
s[j++] = ch;
|
|
}
|
|
r_cons_printf ("%s\n", s);
|
|
free (s);
|
|
}
|
|
}
|
|
break;
|
|
case 'p':
|
|
{
|
|
int mylen = core->block[0];
|
|
// TODO: add support for 2-4 byte length pascal strings
|
|
if (mylen < core->blocksize) {
|
|
r_print_string (core->print, core->offset,
|
|
core->block, mylen, R_PRINT_STRING_ZEROEND);
|
|
core->num->value = mylen;
|
|
} else core->num->value = 0; // error
|
|
}
|
|
break;
|
|
case 'w':
|
|
r_print_string (core->print, core->offset, core->block, len,
|
|
R_PRINT_STRING_WIDE | R_PRINT_STRING_ZEROEND);
|
|
break;
|
|
case ' ':
|
|
len = r_num_math (core->num, input+2);
|
|
r_print_string (core->print, core->offset, core->block, len, 0);
|
|
break;
|
|
default:
|
|
r_print_string (core->print, core->offset, core->block, len,
|
|
R_PRINT_STRING_ZEROEND);
|
|
break;
|
|
}
|
|
break;
|
|
case 'm':
|
|
if (input[1]=='?') {
|
|
r_cons_printf ("Usage: pm [file|directory]\n"
|
|
" r_magic will use given file/dir as reference\n"
|
|
" output of those magic can contain expressions like:\n"
|
|
" foo@0x40 # use 'foo' magic file on address 0x40\n"
|
|
" @0x40 # use current magic file on address 0x40\n"
|
|
" \\n # append newline\n"
|
|
" e dir.magic # defaults to "R_MAGIC_PATH"\n"
|
|
);
|
|
} else r_core_magic (core, input+1, R_TRUE);
|
|
break;
|
|
case 'u':
|
|
r_print_string (core->print, core->offset, core->block, len,
|
|
R_PRINT_STRING_ZEROEND| R_PRINT_STRING_URLENCODE|
|
|
(input[1]=='w')?R_PRINT_STRING_WIDE:0);
|
|
break;
|
|
case 'c':
|
|
r_print_code (core->print, core->offset, core->block, len, input[1]);
|
|
break;
|
|
case 'r':
|
|
r_print_raw (core->print, core->block, len);
|
|
break;
|
|
case 'x':
|
|
{
|
|
int show_offset = r_config_get_i (core->config, "asm.offset");
|
|
if (show_offset) {
|
|
core->print->flags |= R_PRINT_FLAGS_HEADER;
|
|
core->print->flags |= R_PRINT_FLAGS_OFFSET;
|
|
} else {
|
|
core->print->flags &= ~R_PRINT_FLAGS_OFFSET;
|
|
core->print->flags &= ~R_PRINT_FLAGS_HEADER;
|
|
}
|
|
}
|
|
switch (input[1]) {
|
|
case '?':
|
|
eprintf ("Usage: px[owqWQ][f]\n"
|
|
" px show hexdump\n"
|
|
" pxf show hexdump of current function\n"
|
|
" pxo show octal dump\n"
|
|
" pxw show hexadeciaml words dump (32bit)\n"
|
|
" pxW same as above, but one per line\n"
|
|
" pxq show hexadeciaml quad-words dump (64bit)\n"
|
|
" pxQ same as above, but one per line\n"
|
|
);
|
|
break;
|
|
case 'o':
|
|
r_print_hexdump (core->print, core->offset, core->block, len, 8, 1);
|
|
break;
|
|
case 'w':
|
|
r_print_hexdump (core->print, core->offset, core->block, len, 32, 4);
|
|
break;
|
|
case 'W':
|
|
for (i=0; i<len; i+=4) {
|
|
ut32 *p = (ut32*)core->block+i;
|
|
r_cons_printf ("0x%08"PFMT64x" 0x%08x\n", core->offset+i, *p);
|
|
}
|
|
break;
|
|
case 'q':
|
|
r_print_hexdump (core->print, core->offset, core->block, len, 64, 8);
|
|
break;
|
|
case 'Q':
|
|
for (i=0; i<len; i+=8) {
|
|
ut64 *p = (ut64*)core->block+i;
|
|
r_cons_printf ("0x%08"PFMT64x" 0x%016"PFMT64x"\n",
|
|
core->offset+i, *p);
|
|
}
|
|
break;
|
|
default: {
|
|
ut64 from = r_config_get_i (core->config, "diff.from");
|
|
ut64 to = r_config_get_i (core->config, "diff.to");
|
|
if (from == to && from == 0) {
|
|
r_print_hexdump (core->print, core->offset,
|
|
core->block, len, 16, 1);
|
|
} else {
|
|
r_core_print_cmp (core, from, to);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
case '6':
|
|
{
|
|
int malen = (core->blocksize*4)+1;
|
|
ut8 *buf = malloc (malen);
|
|
memset (buf, 0, malen);
|
|
switch (input[1]) {
|
|
case 'e':
|
|
r_base64_encode (buf, core->block, len); //core->blocksize);
|
|
r_cons_printf ("%s\n", buf);
|
|
break;
|
|
case 'd':
|
|
if (r_base64_decode (buf, core->block, len))
|
|
r_cons_printf ("%s\n", buf);
|
|
else eprintf ("r_base64_decode: invalid stream\n");
|
|
break;
|
|
default:
|
|
eprintf ("Usage: p6[ed] [len] base 64 encode/decode\n");
|
|
break;
|
|
}
|
|
free (buf);
|
|
}
|
|
break;
|
|
case '8':
|
|
r_print_bytes (core->print, core->block, len, "%02x");
|
|
break;
|
|
case 'f':
|
|
if (input[1]=='.') {
|
|
if (input[2]=='\0') {
|
|
RListIter *iter;
|
|
RStrHT *sht = core->print->formats;
|
|
int *i;
|
|
r_list_foreach (sht->ls, iter, i) {
|
|
int idx = ((int)(size_t)i)-1;
|
|
const char *key = r_strpool_get (sht->sp, idx);
|
|
const char *val = r_strht_get (core->print->formats, key);
|
|
r_cons_printf ("pf.%s %s\n", key, val);
|
|
}
|
|
} else
|
|
if (input[2]=='-') {
|
|
if (input[3]) r_strht_del (core->print->formats, input+3);
|
|
else r_strht_clear (core->print->formats);
|
|
} else {
|
|
char *name = strdup (input+2);
|
|
char *space = strchr (name, ' ');
|
|
if (space) {
|
|
*space++ = 0;
|
|
//printf ("SET (%s)(%s)\n", name, space);
|
|
r_strht_set (core->print->formats, name, space);
|
|
return 0;
|
|
} else {
|
|
const char *fmt;
|
|
char *eq, *dot = strchr (name, '.');
|
|
if (dot) {
|
|
// TODO: support multiple levels
|
|
*dot++ = 0;
|
|
eq = strchr (dot, '=');
|
|
if (eq) {
|
|
char *res;
|
|
fmt = r_strht_get (core->print->formats, name);
|
|
// TODO: spaguettti, reuse code below.. and handle atoi() too
|
|
res = strdup (fmt);
|
|
*eq++ = 0;
|
|
#if 0
|
|
ut64 v;
|
|
v = r_num_math (NULL, eq);
|
|
r_print_format (core->print, core->offset,
|
|
core->block, core->blocksize, fmt, v, eq);
|
|
#endif
|
|
r_str_word_set0 (res);
|
|
for (i = 1; ; i++) {
|
|
const char *k = r_str_word_get0 (res, i);
|
|
if (!k) break;
|
|
if (!strcmp (k, dot)) {
|
|
r_print_format (core->print, core->offset,
|
|
core->block, core->blocksize, fmt, i-1, eq);
|
|
break;
|
|
}
|
|
}
|
|
free (res);
|
|
} else {
|
|
const char *k, *fmt = r_strht_get (core->print->formats, name);
|
|
if (fmt) {
|
|
if (atoi (dot)>0 || *dot=='0') {
|
|
// indexed field access
|
|
r_print_format (core->print, core->offset,
|
|
core->block, core->blocksize, fmt, atoi (dot), NULL);
|
|
} else {
|
|
char *res = strdup (fmt);
|
|
r_str_word_set0 (res);
|
|
for (i = 1; ; i++) {
|
|
k = r_str_word_get0 (res, i);
|
|
if (!k) break;
|
|
if (!strcmp (k, dot)) {
|
|
r_print_format (core->print, core->offset,
|
|
core->block, core->blocksize, fmt, i-1, NULL);
|
|
break;
|
|
}
|
|
}
|
|
free (res);
|
|
}
|
|
} else {
|
|
|
|
}
|
|
}
|
|
} else {
|
|
const char *fmt = r_strht_get (core->print->formats, name);
|
|
if (fmt) {
|
|
//printf ("GET (%s) = %s\n", name, fmt);
|
|
r_print_format (core->print, core->offset,
|
|
core->block, len, fmt, -1, NULL);
|
|
} else eprintf ("Unknown format (%s)\n", name);
|
|
}
|
|
}
|
|
free (name);
|
|
}
|
|
} else r_print_format (core->print, core->offset,
|
|
core->block, len, input+1, -1, NULL);
|
|
break;
|
|
case 'k':
|
|
{
|
|
char *s = r_print_randomart (core->block, core->blocksize, core->offset);
|
|
r_cons_printf ("%s\n", s);
|
|
free (s);
|
|
}
|
|
break;
|
|
case 'K':
|
|
{
|
|
ut64 offset0 = core->offset;
|
|
r_cons_clear ();
|
|
for (i=0; i< 4; i++) {
|
|
char *s = r_print_randomart (core->block, core->blocksize, core->offset);
|
|
r_cons_gotoxy (0, 0);
|
|
r_cons_printf ("\n%s\n", s);
|
|
free (s);
|
|
core->offset += core->blocksize;
|
|
r_core_read_at (core, core->offset, core->block, core->blocksize);
|
|
s = r_print_randomart (core->block, core->blocksize, core->offset);
|
|
r_cons_printf ("%s\n", s);
|
|
core->offset += core->blocksize;
|
|
r_core_read_at (core, core->offset, core->block, core->blocksize);
|
|
s = r_print_randomart (core->block, core->blocksize, core->offset);
|
|
r_cons_printf ("%s\n", s);
|
|
r_cons_column (i!=3?20:0);
|
|
free (s);
|
|
core->offset += core->blocksize;
|
|
r_core_read_at (core, core->offset, core->block, core->blocksize);
|
|
}
|
|
r_core_read_at (core, offset0, core->block, core->blocksize);
|
|
core->offset = offset0;
|
|
}
|
|
break;
|
|
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;
|
|
case 't':
|
|
switch (input[1]) {
|
|
case ' ':
|
|
case '\0':
|
|
for (l=0; l<len; l+=sizeof (time_t))
|
|
r_print_date_unix (core->print, core->block+l, sizeof (time_t));
|
|
break;
|
|
case 'd':
|
|
for (l=0; l<len; l+=4)
|
|
r_print_date_dos (core->print, core->block+l, 4);
|
|
break;
|
|
case 'n':
|
|
core->print->big_endian = !core->print->big_endian;
|
|
for (l=0; l<len; l+=sizeof (ut64))
|
|
r_print_date_w32 (core->print, core->block+l, sizeof (ut64));
|
|
core->print->big_endian = !core->print->big_endian;
|
|
break;
|
|
case '?':
|
|
r_cons_printf (
|
|
"Usage: pt[dn?]\n"
|
|
" pt print unix time (32 bit cfg.big_endian)\n"
|
|
" ptd print dos time (32 bit cfg.big_endian)\n"
|
|
" ptn print ntfs time (64 bit !cfg.big_endian)\n"
|
|
" pt? show help message\n");
|
|
break;
|
|
}
|
|
break;
|
|
case 'z':
|
|
if (input[1]=='?') {
|
|
r_cons_printf (
|
|
"Usage: pz [len]\n"
|
|
" print N bytes where each byte represents a block of filesize/N\n"
|
|
"Configuration:\n"
|
|
" zoom.maxsz : max size of block\n"
|
|
" zoom.from : start address\n"
|
|
" zoom.to : end address\n"
|
|
" zoom.byte : specify how to calculate each byte\n"
|
|
" p : number of printable chars\n"
|
|
" f : count of flags in block\n"
|
|
" s : strings in range\n"
|
|
" 0 : number of bytes with value '0'\n"
|
|
" F : number of bytes with value 0xFF\n"
|
|
" e : calculate entropy and expand to 0-255 range\n"
|
|
" h : head (first byte value)\n"
|
|
"WARNING: On big files, use 'zoom.byte=h' or restrict ranges\n");
|
|
} else {
|
|
char *oldzoom = NULL;
|
|
ut64 maxsize = r_config_get_i (core->config, "zoom.maxsz");
|
|
ut64 from, to;
|
|
int oldva = core->io->va;
|
|
|
|
from = 0;
|
|
core->io->va = 0;
|
|
to = r_io_size (core->io);
|
|
from = r_config_get_i (core->config, "zoom.from");
|
|
to = r_config_get_i (core->config, "zoom.to");
|
|
if (input[1] && input[1] != ' ') {
|
|
oldzoom = strdup (r_config_get (core->config, "zoom.byte"));
|
|
if (!r_config_set (core->config, "zoom.byte", input+1)) {
|
|
eprintf ("Invalid zoom.byte mode (%s)\n", input+1);
|
|
free (oldzoom);
|
|
return R_FALSE;
|
|
}
|
|
}
|
|
r_print_zoom (core->print, core, printzoomcallback,
|
|
from, to, core->blocksize, (int)maxsize);
|
|
if (oldzoom) {
|
|
r_config_set (core->config, "zoom.byte", oldzoom);
|
|
free (oldzoom);
|
|
}
|
|
if (oldva)
|
|
core->io->va = oldva;
|
|
}
|
|
break;
|
|
default:
|
|
r_cons_printf (
|
|
"Usage: p[=68abcdDfiImrstuxz] [arg|len]\n"
|
|
" p= show entropy bars of full file\n"
|
|
" p6[de] [len] base64 decode/encode\n"
|
|
" p8 [len] 8bit hexpair list of bytes\n"
|
|
" pa [opcode] bytes of assembled opcode\n"
|
|
" pb [len] bitstream of N bytes\n"
|
|
" pc[p] [len] output C (or python) format\n"
|
|
" p[dD][lf] [l] disassemble N opcodes/bytes (see pd?)\n"
|
|
" pf[?|.nam] [fmt] print formatted data (pf.name, pf.name $<expr>) \n"
|
|
" p[iI][f] [len] print N instructions/bytes (f=func) (see pdi)\n"
|
|
" pm [magic] print libmagic data (pm? for more information)\n"
|
|
" pr [len] print N raw bytes\n"
|
|
" p[kK] [len] print key in randomart (K is for mosaic)\n"
|
|
" ps[pwz] [len] print pascal/wide/zero-terminated strings\n"
|
|
" pt[dn?] [len] print different timestamps\n"
|
|
" pu[w] [len] print N url encoded bytes (w=wide)\n"
|
|
" pv[jh] [mode] bar|json|histogram blocks (mode: e?search.in)\n"
|
|
" px[owq] [len] hexdump of N bytes (o=octal, w=32bit, q=64bit)\n"
|
|
" pz [len] print zoom view (see pz? for help)\n");
|
|
break;
|
|
}
|
|
if (tbs != core->blocksize)
|
|
r_core_block_size (core, tbs);
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_hexdump(void *data, const char *input) {
|
|
return cmd_print (data, input-1);
|
|
}
|