2011-02-12 01:52:41 +01:00
|
|
|
/* radare - LGPL - Copyright 2007-2011 pancake<nopcode.org> */
|
2009-02-05 22:08:46 +01:00
|
|
|
|
|
|
|
#include "r_cons.h"
|
2009-03-11 11:42:11 +00:00
|
|
|
#include "r_util.h"
|
2009-02-05 22:08:46 +01:00
|
|
|
#include "r_print.h"
|
|
|
|
|
2011-03-04 13:08:40 +01:00
|
|
|
static void print_format_help(RPrint *p) {
|
2010-03-01 10:49:04 +01:00
|
|
|
p->printf (
|
2011-03-04 13:08:40 +01:00
|
|
|
"Usage: pf [times][format] [arg0 arg1]\n"
|
|
|
|
"Example: pf 10xiz pointer length string\n"
|
|
|
|
"Example: pf {array_size}b @ array_base\n"
|
2009-03-11 11:42:11 +00:00
|
|
|
" e - temporally swap endian\n"
|
|
|
|
//" D - double (8 bytes)\n"
|
|
|
|
" f - float value\n"
|
|
|
|
" b - one byte \n"
|
2011-02-12 12:54:26 +01:00
|
|
|
" B - show 10 first bytes of buffer\n" // B must be for binary ??
|
2010-06-25 11:22:14 +02:00
|
|
|
" i - %%i integer value (4 bytes)\n"
|
2009-03-11 11:42:11 +00:00
|
|
|
" w - word (16 bit hexa)\n"
|
|
|
|
" q - quadword (8 bytes)\n"
|
|
|
|
" p - pointer reference\n"
|
2010-06-25 11:22:14 +02:00
|
|
|
" d - 0x%%08x hexadecimal value\n"
|
2009-03-11 11:42:11 +00:00
|
|
|
" X - 0x%%08x hexadecimal value and flag (fd @ addr)\n"
|
|
|
|
" z - \\0 terminated string\n"
|
|
|
|
" Z - \\0 terminated wide string\n"
|
|
|
|
" s - pointer to string\n"
|
|
|
|
" t - unix timestamp string\n"
|
|
|
|
" * - next char is pointer\n"
|
|
|
|
" . - skip 1 byte\n");
|
|
|
|
}
|
|
|
|
|
2009-03-12 00:42:32 +00:00
|
|
|
/* TODO: needs refactoring */
|
2010-03-01 10:49:04 +01:00
|
|
|
R_API void r_print_format(struct r_print_t *p, ut64 seek, const ut8* buf, int len, const char *fmt) {
|
2009-02-05 22:08:46 +01:00
|
|
|
unsigned char buffer[256];
|
2011-02-12 01:52:41 +01:00
|
|
|
int nargs, i, j, idx, times, otimes, endian;
|
|
|
|
char *args, *bracket, tmp, last = 0;
|
2009-02-05 22:08:46 +01:00
|
|
|
const char *arg = fmt;
|
2009-03-12 00:42:32 +00:00
|
|
|
const char *argend = arg+strlen(fmt);
|
2009-07-08 13:49:55 +02:00
|
|
|
ut64 addr = 0;
|
2009-03-12 00:42:32 +00:00
|
|
|
char namefmt[8];
|
2011-02-12 01:52:41 +01:00
|
|
|
nargs = endian = i = j = 0;
|
2009-02-05 22:08:46 +01:00
|
|
|
|
2011-02-12 01:52:41 +01:00
|
|
|
while (*arg && *arg==' ') arg++;
|
2009-02-05 22:08:46 +01:00
|
|
|
/* get times */
|
2011-02-12 01:52:41 +01:00
|
|
|
otimes = times = atoi (arg);
|
2009-02-05 22:08:46 +01:00
|
|
|
if (times > 0)
|
2011-02-12 01:52:41 +01:00
|
|
|
while ((*arg>='0'&&*arg<='9')) arg++;
|
2010-03-01 10:49:04 +01:00
|
|
|
bracket = strchr (arg,'{');
|
2009-02-05 22:08:46 +01:00
|
|
|
if (bracket) {
|
2010-03-01 10:49:04 +01:00
|
|
|
char *end = strchr (arg,'}');
|
2009-02-05 22:08:46 +01:00
|
|
|
if (end == NULL) {
|
2010-03-01 10:49:04 +01:00
|
|
|
eprintf ("No end bracket. Try pm {ecx}b @ esi\n");
|
2009-02-05 22:08:46 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
*end='\0';
|
2010-03-01 10:49:04 +01:00
|
|
|
times = r_num_math (NULL, bracket+1);
|
2009-03-11 11:42:11 +00:00
|
|
|
arg = end + 1;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (arg[0]=='\0') {
|
2011-03-04 13:08:40 +01:00
|
|
|
print_format_help (p);
|
2009-02-05 22:08:46 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* get args */
|
2010-03-01 10:49:04 +01:00
|
|
|
args = strchr (arg, ' ');
|
2009-02-05 22:08:46 +01:00
|
|
|
if (args) {
|
2009-09-25 04:04:51 +02:00
|
|
|
int l=0, maxl = 0;
|
2009-03-12 00:42:32 +00:00
|
|
|
argend = args;
|
2010-03-01 10:49:04 +01:00
|
|
|
args = strdup (args+1);
|
|
|
|
nargs = r_str_word_set0 (args+1);
|
2009-02-05 22:08:46 +01:00
|
|
|
if (nargs == 0)
|
2010-03-01 10:49:04 +01:00
|
|
|
R_FREE (args);
|
|
|
|
for (i=0;i<nargs;i++) {
|
|
|
|
int len = strlen (r_str_word_get0 (args+1, i));
|
2009-09-25 04:04:51 +02:00
|
|
|
if (len>maxl) maxl = len;
|
2009-03-12 00:42:32 +00:00
|
|
|
}
|
|
|
|
l++;
|
2010-03-01 10:49:04 +01:00
|
|
|
sprintf (namefmt, "%%%ds : ", maxl);
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* go format */
|
|
|
|
i = 0;
|
2010-03-01 10:49:04 +01:00
|
|
|
if (times==0)
|
|
|
|
otimes = times = 1;
|
2009-03-11 11:42:11 +00:00
|
|
|
for(;times;times--) { // repeat N times
|
2009-02-05 22:08:46 +01:00
|
|
|
const char * orig = arg;
|
|
|
|
if (otimes>1)
|
2010-04-14 13:02:23 +02:00
|
|
|
p->printf ("0x%08"PFMT64x" [%d] {\n", seek+i, otimes-times);
|
2009-03-12 00:42:32 +00:00
|
|
|
for(idx=0;arg<argend && idx<len;idx++, arg=arg+1) {
|
2009-02-05 22:08:46 +01:00
|
|
|
addr = 0LL;
|
|
|
|
if (endian)
|
|
|
|
addr = (*(buf+i))<<24 | (*(buf+i+1))<<16 | *(buf+i+2)<<8 | *(buf+i+3);
|
|
|
|
else addr = (*(buf+i+3))<<24 | (*(buf+i+2))<<16 | *(buf+i+1)<<8 | *(buf+i);
|
|
|
|
|
|
|
|
tmp = *arg;
|
|
|
|
feed_me_again:
|
|
|
|
if (tmp == 0 && last != '*')
|
|
|
|
break;
|
|
|
|
/* skip chars */
|
|
|
|
switch(tmp) {
|
|
|
|
case '*':
|
|
|
|
if (i<=0) break;
|
|
|
|
tmp = last;
|
|
|
|
arg = arg - 1;
|
|
|
|
idx--;
|
|
|
|
goto feed_me_again;
|
|
|
|
case 'e': // tmp swap endian
|
|
|
|
idx--;
|
|
|
|
endian ^=1;
|
|
|
|
continue;
|
|
|
|
case '.': // skip char
|
|
|
|
i++;
|
|
|
|
idx--;
|
|
|
|
continue;
|
|
|
|
case '?': // help
|
2011-03-04 13:08:40 +01:00
|
|
|
print_format_help (p);
|
2009-02-05 22:08:46 +01:00
|
|
|
idx--;
|
|
|
|
i=len; // exit
|
|
|
|
continue;
|
|
|
|
}
|
2010-03-01 10:49:04 +01:00
|
|
|
if (idx<nargs)
|
|
|
|
p->printf (namefmt, r_str_word_get0(args, idx));
|
2009-02-05 22:08:46 +01:00
|
|
|
/* cmt chars */
|
2011-03-04 13:08:40 +01:00
|
|
|
switch (tmp) {
|
2009-02-05 22:08:46 +01:00
|
|
|
#if 0
|
|
|
|
case 'n': // enable newline
|
|
|
|
j ^= 1;
|
|
|
|
continue;
|
|
|
|
#endif
|
2009-03-11 11:42:11 +00:00
|
|
|
#if 0
|
2009-02-05 22:08:46 +01:00
|
|
|
case 't':
|
|
|
|
/* unix timestamp */
|
|
|
|
D cons_printf("0x%08x = ", config.seek+i);
|
|
|
|
{
|
|
|
|
/* dirty hack */
|
|
|
|
int oldfmt = last_print_format;
|
2009-07-08 13:49:55 +02:00
|
|
|
ut64 old = config.seek;
|
2009-02-05 22:08:46 +01:00
|
|
|
radare_seek(config.seek+i, SEEK_SET);
|
|
|
|
radare_read(0);
|
|
|
|
print_data(config.seek+i, "8", buf+i, 4, FMT_TIME_UNIX);
|
|
|
|
last_print_format=oldfmt;
|
|
|
|
radare_seek(old, SEEK_SET);
|
|
|
|
}
|
|
|
|
break;
|
2009-03-11 11:42:11 +00:00
|
|
|
#endif
|
2009-02-05 22:08:46 +01:00
|
|
|
case 'e': {
|
|
|
|
double doub;
|
2011-02-12 01:52:41 +01:00
|
|
|
memcpy (&doub, buf+i, sizeof (double));
|
|
|
|
p->printf ("%e = ", doub);
|
|
|
|
p->printf ("(double)");
|
2009-02-05 22:08:46 +01:00
|
|
|
i+=8;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'q':
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
|
|
|
p->printf ("(qword)");
|
2009-02-05 22:08:46 +01:00
|
|
|
i+=8;
|
|
|
|
break;
|
|
|
|
case 'b':
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
|
|
|
p->printf ("%d ; 0x%02x ; '%c' ",
|
2009-03-11 11:42:11 +00:00
|
|
|
buf[i], buf[i], IS_PRINTABLE(buf[i])?buf[i]:0);
|
2009-02-05 22:08:46 +01:00
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
case 'B':
|
|
|
|
memset(buffer, '\0', 255);
|
2011-02-17 00:58:54 +01:00
|
|
|
if (!p->iob.read_at) {
|
2011-02-12 01:52:41 +01:00
|
|
|
printf ("(cannot read memory)\n");
|
2009-03-11 11:42:11 +00:00
|
|
|
break;
|
2011-02-17 00:58:54 +01:00
|
|
|
} else p->iob.read_at (p->iob.io, (ut64)addr, buffer, 248);
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
|
|
|
for (j=0;j<10;j++) p->printf ("%02x ", buf[j]);
|
2009-03-12 00:42:32 +00:00
|
|
|
p->printf(" ... (");
|
2011-02-12 01:52:41 +01:00
|
|
|
for (j=0;j<10;j++)
|
|
|
|
if (IS_PRINTABLE (buf[j]))
|
2009-03-12 00:42:32 +00:00
|
|
|
p->printf("%c", buf[j]);
|
|
|
|
p->printf(")");
|
2009-02-05 22:08:46 +01:00
|
|
|
i+=4;
|
|
|
|
break;
|
2010-06-25 11:22:14 +02:00
|
|
|
case 'i':
|
2009-03-12 00:42:32 +00:00
|
|
|
p->printf("0x%08x = ", seek+i);
|
|
|
|
p->printf("%d", addr);
|
2009-02-05 22:08:46 +01:00
|
|
|
i+=4;
|
|
|
|
break;
|
2010-06-25 11:22:14 +02:00
|
|
|
case 'd':
|
2009-03-12 00:42:32 +00:00
|
|
|
p->printf("0x%08x = ", seek+i);
|
|
|
|
p->printf("0x%08x ", addr);
|
2009-02-05 22:08:46 +01:00
|
|
|
i+=4;
|
|
|
|
break;
|
|
|
|
case 'X': {
|
2009-07-08 13:49:55 +02:00
|
|
|
ut32 addr32 = (ut32)addr;
|
2009-03-11 11:42:11 +00:00
|
|
|
//char buf[128];
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
|
|
|
p->printf ("0x%08"PFMT64x" ", addr32);
|
2009-07-08 13:49:55 +02:00
|
|
|
//if (string_flag_offset(buf, (ut64)addr32, -1))
|
2009-03-12 00:42:32 +00:00
|
|
|
// p->printf("; %s", buf);
|
2009-02-05 22:08:46 +01:00
|
|
|
i+=4;
|
|
|
|
} break;
|
|
|
|
case 'w':
|
|
|
|
case '1': // word (16 bits)
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
2009-02-05 22:08:46 +01:00
|
|
|
if (endian)
|
|
|
|
addr = (*(buf+i))<<8 | (*(buf+i+1));
|
|
|
|
else addr = (*(buf+i+1))<<8 | (*(buf+i));
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%04x ", addr);
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
|
|
|
case 'z': // zero terminated string
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
|
|
|
for (;buf[i]&&i<len; i++) {
|
|
|
|
if (IS_PRINTABLE (buf[i]))
|
|
|
|
p->printf ("%c", buf[i]);
|
|
|
|
else p->printf (".");
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'Z': // zero terminated wide string
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
|
|
|
for (;buf[i]&&i<len; i+=2) {
|
|
|
|
if (IS_PRINTABLE (buf[i]))
|
|
|
|
p->printf ("%c", buf[i]);
|
|
|
|
else p->printf (".");
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf (" ");
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
|
|
|
case 's':
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x = ", seek+i);
|
|
|
|
memset (buffer, '\0', 255);
|
2011-02-17 00:58:54 +01:00
|
|
|
if (p->iob.read_at)
|
|
|
|
p->iob.read_at (p->iob.io, (ut64)addr, buffer, 248);
|
2009-03-11 11:42:11 +00:00
|
|
|
else {
|
2011-02-12 01:52:41 +01:00
|
|
|
printf ("(cannot read memory)\n");
|
2009-03-11 11:42:11 +00:00
|
|
|
break;
|
|
|
|
}
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("0x%08x -> 0x%08x ", seek+i, addr);
|
|
|
|
p->printf ("%s ", buffer);
|
2009-02-05 22:08:46 +01:00
|
|
|
i+=4;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* ignore unknown chars */
|
2009-03-12 00:42:32 +00:00
|
|
|
break;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("\n");
|
2009-02-05 22:08:46 +01:00
|
|
|
last = tmp;
|
|
|
|
}
|
|
|
|
if (otimes>1)
|
2011-02-12 01:52:41 +01:00
|
|
|
p->printf ("}\n");
|
2009-02-05 22:08:46 +01:00
|
|
|
arg = orig;
|
2009-03-11 11:42:11 +00:00
|
|
|
idx = 0;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2009-03-11 11:42:11 +00:00
|
|
|
// free((void *)&args);
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|