radare2/libr/util/p_format.c
2014-11-01 22:12:05 +01:00

1145 lines
28 KiB
C

/* radare - LGPL - Copyright 2007-2014 - pancake */
#include "r_cons.h"
#include "r_util.h"
#include "r_print.h"
#define NOPTR 0
#define PTRSEEK 1
#define PTRBACK 2
#define NULLPTR 3
#define STRUCTPTR 100
#define NESTEDSTRUCT 1
#define STRUCTFLAG 10000
#define NESTDEPTH 14
static int (*realprintf)(const char *str, ...);
static int nullprintf(const char *fmt, ...) { return 0; }
static void updateAddr(const ut8 *buf, int i, int endian, ut64 *addr, ut64 *addr64) {
if (addr) {
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));
}
if (addr64) {
if (endian)
*addr64 = (((ut64)(*(buf+i))<<56))
| ((ut64)(*(buf+i+1))<<48)
| ((ut64)(*(buf+i+2))<<40)
| ((ut64)(*(buf+i+3))<<32)
| ((ut64)(*(buf+i+4))<<24)
| ((ut64)(*(buf+i+5))<<16)
| ((ut64)(*(buf+i+6))<<8)
| ((ut64)(*(buf+i+7)));
else
*addr64 =(((ut64)(*(buf+i+7))<<56))
| ((ut64)(*(buf+i+6))<<48)
| ((ut64)(*(buf+i+5))<<40)
| ((ut64)(*(buf+i+4))<<32)
| ((ut64)(*(buf+i+3))<<24)
| ((ut64)(*(buf+i+2))<<16)
| ((ut64)(*(buf+i+1))<<8)
| ((ut64)(*(buf+i)));
}
}
static void r_print_format_quadword(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
ut64 addr64;
updateAddr (buf, i, endian, NULL, &addr64);
if (mustset) {
realprintf ("wv8 %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
if (!json) {
p->printf ("0x%08"PFMT64x" = ", seeki);
p->printf ("(qword) ");
}
if (size==-1)
if (json)
p->printf ("%d", addr64);
else
p->printf ("0x%016"PFMT64x, addr64);
else {
if (json)
p->printf ("[ %d", addr64);
else
p->printf ("[ 0x%016"PFMT64x, addr64);
size--;
i+=8;
while (size--) {
updateAddr (buf, i, endian, NULL, &addr64);
if (json)
p->printf (", %d", addr64);
else
p->printf (", 0x%016"PFMT64x, addr64);
i+=8;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
static void r_print_format_byte(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
if (mustset) {
realprintf ("\"w %s\" @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (size==-1)
if (json)
p->printf ("%d", buf[i]);
else
p->printf ("%d ; 0x%02x ; '%c'", buf[i], buf[i],
IS_PRINTABLE (buf[i])?buf[i]:'.');
else {
if (json)
p->printf ("[ %d", buf[i]);
else
p->printf ("[ %d ; 0x%02x ; '%c'", buf[i], buf[i],
IS_PRINTABLE (buf[i])?buf[i]:'.');
size--;
i++;
while (size--) {
if (json)
p->printf (", %d", buf[i]);
else
p->printf (", %d ; 0x%02x ; '%c'", buf[i], buf[i],
IS_PRINTABLE (buf[i])?buf[i]:'.');
i++;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
static void r_print_format_char(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
if (mustset) {
realprintf ("\"w %s\" @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (size==-1)
if (json)
p->printf ("\"%c\"", buf[i]);
else
p->printf (" %d ; '%c'", buf[i], buf[i], buf[i],
IS_PRINTABLE (buf[i])?buf[i]:'.');
else {
if (json)
p->printf ("[ \"%c\"", buf[i]);
else
p->printf ("[ %d ; '%c'", buf[i], buf[i], buf[i],
IS_PRINTABLE (buf[i])?buf[i]:'.');
size--;
i++;
while (size--) {
if (json)
p->printf (", \"%c\"", buf[i]);
else
p->printf (", %d ; '%c'", buf[i], buf[i], buf[i],
IS_PRINTABLE (buf[i])?buf[i]:'.');
i++;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
static int r_print_format_string(const RPrint* p, ut64 seeki, ut64 addr64, ut64 addr, int is64, int json) {
ut8 buffer[255];
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (p->iob.read_at) {
if (is64 == 1)
p->iob.read_at (p->iob.io, addr64, buffer, sizeof (buffer)-8);
else
p->iob.read_at (p->iob.io, (ut64)addr, buffer, sizeof (buffer)-8);
} else {
printf ("(cannot read memory)\n");
return -1;
}
if (json)
p->printf ("%d,\"string\":\"%s\"}", seeki, addr);
else {
p->printf ("0x%08"PFMT64x" -> 0x%08"PFMT64x" ", seeki, addr);
p->printf ("%s", buffer);
}
return 0;
}
static void r_print_format_time(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
ut64 addr;
updateAddr (buf, i, endian, &addr, NULL);
if (mustset) {
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
char *time = strdup(ctime((time_t*)&addr));
*(time+24) = '\0';
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (size==-1)
if (json)
p->printf ("\"%s\"", time);
else
p->printf ("%s", time);
else {
if (json)
p->printf ("[ \"%s\"", time);
else
p->printf ("[ %s", time);
size--;
i+=4;
while (size--) {
updateAddr (buf, i, endian, &addr, NULL);
time = strdup(ctime((time_t*)&addr));
*(time+24) = '\0';
if (json)
p->printf (", \"%s\"", time);
else
p->printf (", %s", time);
i+=4;
}
p->printf (" ]");
}
free (time);
if (json) p->printf ("}");
}
}
// TODO: support unsigned int?
static void r_print_format_hex(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
ut64 addr;
updateAddr (buf, i, endian, &addr, NULL);
if (mustset) {
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (size==-1)
if (json)
p->printf ("%d", addr);
else
p->printf ("%"PFMT64d, addr);
else {
if (json)
p->printf ("[ %d", addr);
else
p->printf ("[ %"PFMT64d, addr);
size--;
i+=4;
while (size--) {
updateAddr (buf, i, endian, &addr, NULL);
if (json)
p->printf (", %d", addr);
else
p->printf (", %"PFMT64d, addr);
i+=4;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
static void r_print_format_octal (const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
ut64 addr;
updateAddr (buf, i, endian, &addr, NULL);
if (mustset) {
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
ut32 addr32 = (ut32)addr;
if (!json) {
p->printf ("0x%08"PFMT64x" = ", seeki);
p->printf ("(octal) ");
}
if (size==-1)
if (json)
p->printf ("%d", addr32);
else
p->printf ("0%08"PFMT64o, addr32);
else {
if (json)
p->printf ("[ %d", addr32);
else
p->printf ("[ 0%08"PFMT64o, addr32);
size--;
i+=4;
while (size--) {
updateAddr (buf, i, endian, &addr, NULL);
addr32 = (ut32)addr;
if (json)
p->printf (", %d", addr32);
else
p->printf (", 0%08"PFMT64o, addr32);
i+=4;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
static void r_print_format_hexflag(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
ut64 addr;
updateAddr (buf, i, endian, &addr, NULL);
if (mustset) {
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
ut32 addr32 = (ut32)addr;
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (size==-1)
if (json)
p->printf ("%d", addr32);
else
p->printf ("0x%08"PFMT64x, addr32);
else {
if (json)
p->printf ("[ %d", addr32);
else
p->printf ("[ 0x%08"PFMT64x, addr32);
size--;
i+=4;
while (size--) {
updateAddr (buf, i, endian, &addr, NULL);
addr32 = (ut32)addr;
if (json)
p->printf (",%d", addr32);
else
p->printf (", 0x%08"PFMT64x, addr32);
i+=4;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
static int r_print_format_10bytes(const RPrint* p, int mustset, const char* setval, ut64 seeki, ut64 addr, ut8* buf, int json) {
ut8 buffer[255];
int j;
if (mustset) {
realprintf ("?e pf B not yet implemented\n");
} else {
if (!p->iob.read_at) {
printf ("(cannot read memory)\n");
return -1;
} else
p->iob.read_at (p->iob.io, (ut64)addr, buffer, 248);
if (!json) {
p->printf ("0x%08"PFMT64x" = ", seeki);
j=0;
} else {
p->printf ("[ %d", buf[0]);
j=1;
}
for (; j<10; j++)
if (json)
p->printf (", %d", buf[j]);
else
p->printf ("%02x ", buf[j]);
if (!json)
p->printf (" ... (");
else {
p->printf ("]}");
return 0;
}
for (j=0; j<10; j++)
if (IS_PRINTABLE (buf[j]))
p->printf ("%c", buf[j]);
else
p->printf (".");
p->printf (")");
}
return 0;
}
static int r_print_format_hexpairs(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int size, int json) {
int j;
if (mustset) {
realprintf ("?e pf X not yet implemented\n");
} else {
size = (size < 1) ? 1 : size;
if (!json) {
p->printf ("0x%08"PFMT64x" = ", seeki);
j=0;
} else {
p->printf ("[ %d", buf[0]);
j=1;
}
for (; j<10; j++)
if (json)
p->printf (", %d", buf[j]);
else
p->printf ("%02x ", buf[j]);
if (!json)
p->printf (" ... (");
else {
p->printf ("]}");
return size;
}
p->printf (" ... (");
for (j=0; j<size; j++)
if (IS_PRINTABLE (buf[j]))
p->printf ("%c", buf[j]);
else
p->printf (".");
p->printf (")");
}
return size;
}
static void r_print_format_float(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
ut64 addr;
updateAddr (buf, i, endian, &addr, NULL);
if (mustset) {
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (size==-1)
p->printf ("%f", (float)addr);
else {
p->printf ("[ %f", (float)addr);
size--;
i+=4;
while (size--) {
updateAddr (buf, i, endian, &addr, NULL);
p->printf (", %f", (float)addr);
i+=4;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
static void r_print_format_word(const RPrint* p, int endian, int mustset,
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
ut64 addr;
if (endian)
addr = (*(buf+i))<<8 | (*(buf+i+1));
else addr = (*(buf+i+1))<<8 | (*(buf+i));
if (mustset) {
realprintf ("wv2 %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (size==-1)
if (json)
p->printf ("%d", addr);
else
p->printf ("0x%04x", addr);
else {
if (json)
p->printf ("[ %d", addr);
else
p->printf ("[ 0x%04x", addr);
size--;
i+=2;
while (size--) {
if (endian)
addr = (*(buf+i))<<8 | (*(buf+i+1));
else addr = (*(buf+i+1))<<8 | (*(buf+i));
if (json)
p->printf (", %d", addr);
else
p->printf (", 0x%04x", addr);
i+=2;
}
p->printf (" ]");
}
if (json) p->printf ("}");
}
}
// XXX: this is very incomplete. must be updated to handle all format chars
int r_print_format_struct_size(const char *f, RPrint *p) {
char *o = strdup(f);
char *end = strchr (o, ' '), *args, *fmt = o;
int size = 0, tabsize=0, i, idx=0;
if (!end && !(end = strchr (o, '\0')))
return -1;
*end = 0;
args = strdup (end+1);
r_str_word_set0 (args);
for (i=0; i<strlen (fmt); i++) {
if (fmt[i] == '[') {
char *end = strchr (fmt+i,']');
if (end == NULL) {
eprintf ("No end bracket.\n");
continue;
}
*end = '\0';
tabsize = r_num_math (NULL, fmt+i+1);
*end = ']';
while (fmt[i++]!=']');
} else {
tabsize = 1;
}
switch (fmt[i]) {
case 'c':
case 'b':
case '.':
size+=tabsize*1;
break;
case 'w':
size += tabsize*2;
break;
case 'd':
case 'o':
case 'i':
case 'x':
case 'f':
case 's':
case 't':
case ':':
size += tabsize*4;
break;
case 'S':
case 'q':
size += tabsize*8;
break;
case 'z':
case 'Z':
size += tabsize;
break;
case '*':
size += tabsize*4;
i++;
break;
case 'B':
case 'E':
switch (tabsize) {
case 1: size+=1; break;
case 2: size+=2; break;
case 4: size+=4; break;
default: break;
}
break;
case '?':
{
const char *format = NULL;
char *endname = NULL, *structname = NULL;
structname = strdup(r_str_word_get0 (args, idx));
if (*structname == '(') {
endname = strchr (structname, ')');
} else {
eprintf ("Struct name missing (%s)\n", structname);
free(structname);
break;
}
if (endname) *endname = '\0';
format = r_strht_get (p->formats, structname+1);
size += tabsize * r_print_format_struct_size (format, p);
free (structname);
break;
}
// TODO continue list
default:
break;
}
idx++;
}
free (o);
free (args);
return size;
}
static int r_print_format_struct(RPrint* p, ut64 seek, const ut8* b, int len, char *name, int slide, int json) {
const char *fmt;
int flag = (slide>=STRUCTFLAG)?SEEFLAG:-1;
flag = (json)?JSONOUTPUT:flag;
if ((slide%STRUCTPTR) > NESTDEPTH) {
eprintf ("Too much nested struct, recursion too deep...\n");
return 0;
}
if (flag) p->printf = realprintf;
fmt = r_strht_get (p->formats, name);
if (!fmt || !*fmt) {
eprintf ("Undefined struct '%s'.\n", name);
return 0;
}
r_print_format (p, seek, b, len, fmt, flag, NULL);
return r_print_format_struct_size(fmt, p);
}
R_API int r_print_format(RPrint *p, ut64 seek, const ut8* b, const int len,
const char *fmt, int elem, const char *setval) {
int nargs, i, j, invalid, nexti, idx, times, otimes, endian, isptr = 0;
int (*oldprintf)(const char *str, ...);
const char *argend = fmt+strlen (fmt);
ut64 addr = 0, addr64 = 0, seeki = 0;;
char *args = NULL, *bracket, tmp, last = 0;
const char *arg = fmt;
int viewflags = 0, flag = (elem==SEEFLAG)?1:0, json = (elem==JSONOUTPUT)?1:0;
char namefmt[8];
static int slide=0, oldslide=0;
ut8 *buf;
nexti = nargs = i = j = 0;
if (len < 1)
return 0;
buf = malloc (len);
if (!buf)
return 0;
memcpy (buf, b, len);
endian = p->big_endian;
oldprintf = NULL;
realprintf = p->printf;
while (*arg && iswhitechar (*arg)) arg++;
/* get times */
otimes = times = atoi (arg);
if (times > 0)
while ((*arg>='0'&&*arg<='9')) arg++;
bracket = strchr (arg,'{');
if (bracket) {
char *end = strchr (arg, '}');
if (end == NULL) {
eprintf ("No end bracket. Try pm {ecx}b @ esi\n");
goto beach;
}
*end='\0';
times = r_num_math (NULL, bracket+1);
arg = end + 1;
}
if (*arg=='\0') {
goto beach;
}
/* get args */
args = strchr (arg, ' ');
if (args) {
int l=0, maxl = 0;
argend = args;
args = strdup (args+1);
nargs = r_str_word_set0 (args);
if (nargs == 0)
R_FREE (args);
for (i=0; i<nargs; i++) {
const int len = strlen (r_str_word_get0 (args, i));
if (len > maxl)
maxl = len;
}
l++;
snprintf (namefmt, sizeof (namefmt), "%%%ds : ", maxl+6*slide%STRUCTPTR);
}
#define ISPOINTED ((slide%STRUCTFLAG)/STRUCTPTR<=(oldslide%STRUCTFLAG)/STRUCTPTR)
#define ISNESTED ((slide%STRUCTPTR)<=(oldslide%STRUCTPTR))
if (json && slide==0) p->printf("[");
/* go format */
i = 0;
if (!times)
otimes = times = 1;
for (; times; times--) { // repeat N times
const char * orig = arg;
int first = 1;
if (otimes>1) {
if (json) {
if (otimes > times) p->printf (",");
p->printf ("[{\"index\":%d,\"offset\":%d},", otimes-times, seek+i);
} else
p->printf ("0x%08"PFMT64x" [%d] {\n", seek+i, otimes-times);
}
arg = orig;
for (idx=0; i<len && arg<argend && *arg; arg++) {
int size;
char *name = NULL;
seeki = seek+i;
addr = 0LL;
invalid = 0;
if (arg[0] == '[') {
char *end = strchr (arg,']');
if (end == NULL) {
eprintf ("No end bracket.\n");
goto beach;
}
*end = '\0';
size = r_num_math (NULL, arg+1);
arg = end + 1;
*end = ']';
} else {
size = -1;
}
updateAddr (buf, i, endian, &addr, &addr64);
tmp = *arg;
if (!json && otimes>1)
p->printf (" ");
#define MUSTSET (setval && elem == idx)
#define MUSTSEE (elem == -1 || elem == idx)
#define ISSTRUCT (tmp == '?' || (tmp == '*' && *(arg+1) == '?'))
if ((MUSTSEE && !flag)) {
if (!(MUSTSET)) {
if (oldprintf)
p->printf = oldprintf;
if (idx<nargs && tmp != 'e' && isptr == 0) {
const char *name = r_str_word_get0 (args, idx);
if (ISSTRUCT || tmp=='E' || tmp=='B') {
if (*name == '(') {
name = strchr (name, ')')+1;
} else {
eprintf ("Missing name (%s)\n", name);
goto beach;
}
}
p->printf (namefmt, name);
idx++;
}
}
} else {
if (!oldprintf)
oldprintf = p->printf;
p->printf = nullprintf;
}
feed_me_again:
switch (isptr) {
case 1:
{
nexti = i + (p->bits/8);
i = 0;
if(tmp == '?' )seeki = addr;
memset (buf, '\0', len);
if (!json)
p->printf ("(*0x%"PFMT64x") ", addr);
if (addr == 0) isptr = NULLPTR;
else isptr = PTRBACK;
if (/*addr<(b+len) && addr>=b && */p->iob.read_at) { /* The test was here to avoid segfault in the next line,
but len make it doesnt work... */
p->iob.read_at (p->iob.io, (ut64)addr, buf, len-4);
updateAddr (buf, i, endian, &addr, &addr64);
} else {
eprintf ("(SEGFAULT: cannot read memory at 0x%08"PFMT64x", Block: %s, blocksize: 0x%x)\n",
addr, b, len);
p->printf("\n");
goto beach;
}
}
break;
case 2:
// restore state after pointer seek
i = nexti;
seeki = seek+i;
memcpy (buf, b, len);
isptr = NOPTR;
arg--;
continue;
}
if (tmp == 0 && last != '*')
break;
/* skip chars */
switch (tmp) {
case '*': // next char is a pointer
isptr = PTRSEEK;
arg++;
tmp = *arg; //last;
goto feed_me_again;
case '+': // toggle view flags
viewflags = !viewflags;
continue;
case 'e': // tmp swap endian
endian ^= 1;
continue;
case ':': // skip 4 bytes
if (size == -1) i+=4;
else
while (size--) i+=4;
continue;
case '.': // skip 1 byte
if (size == -1) i++;
else
i+=size;
continue;
case 'p': // pointer reference
tmp = (p->bits == 64)? 'q': 'x';
break;
}
if (flag && isptr != NULLPTR) {
if (tmp == '?') {
char *n = strdup (r_str_word_get0 (args, idx)+1);
char *par = strchr (n, ')');
if (par == NULL) {
eprintf ("No end parenthesis for struct name");
free (n);
goto beach;
} else {
*par = '.';
}
realprintf ("f %s_", n);
free(n);
} else if (tmp == 'E') {
char *n = strdup (r_str_word_get0 (args, idx)+1);
char *par = strchr (n, ')');
if (par == NULL) {
eprintf ("No end parenthesis for enum name");
free (n);
goto beach;
}
par++;
realprintf ("f %s=0x%08"PFMT64x"\n", par, seeki);
free(n);
} else if (slide>0 && idx==0) {
realprintf ("%s=0x%08"PFMT64x"\n",
r_str_word_get0 (args, idx), seeki);
} else realprintf ("f %s=0x%08"PFMT64x"\n",
r_str_word_get0 (args, idx) , seeki);
idx++;
}
if (json) {
char *structname, *osn;
if (oldprintf)
p->printf = oldprintf;
structname = osn = strdup (r_str_word_get0 (args, idx++));
if (ISSTRUCT) {
if (*structname == '(') {
name = strchr (structname, ')');
} else {
eprintf ("Struct name missing (%s)\n", structname);
free (structname);
goto beach;
}
structname++;
if (name) *(name++) = '\0';
else eprintf ("No ')'\n");
} else {
name = osn;
}
if (oldslide<=slide) {
if (!first)
p->printf (",");
else
first = 0;
} else if(oldslide!=0) {
p->printf ("]},");
}
p->printf ("{\"name\":\"%s\",\"type\":\"", name);
if (ISSTRUCT) {
p->printf ("%s", structname);
} else {
p->printf ("%c", tmp);
}
if (isptr) p->printf ("*");
p->printf ("\",\"offset\":%d,\"value\":",(isptr)?(seek+nexti-(p->bits/8)):seek+i);
free (osn);
}
if (isptr == NULLPTR) {
if (json)
p->printf ("\"NULL\"}", tmp, seek+i);
else
p->printf ("NULL");
isptr = PTRBACK;
} else
/* format chars */
switch (tmp) {
case 't':
r_print_format_time(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i+= (size==-1) ? 4 : 4*size;
break;
case 'q':
r_print_format_quadword(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i += (size==-1) ? 8 : 8*size;
break;
case 'b':
r_print_format_byte(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i+= (size==-1) ? 1 : size;
break;
case 'c':
r_print_format_char (p, endian, MUSTSET,
setval, seeki, buf, i, size, json);
i+= (size==-1) ? 1 : size;
break;
case 'X':
size = r_print_format_hexpairs (p, endian, MUSTSET,
setval, seeki, buf, size, json);
i += size;
break;
case 'T':
if(r_print_format_10bytes(p, MUSTSET,
setval, seeki, addr, buf, json) == 0)
i += (size==-1) ? 4 : 4*size;
break;
case 'f':
r_print_format_float(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i += (size==-1) ? 4 : 4*size;
break;
case 'i':
case 'd':
r_print_format_hex(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i+= (size==-1) ? 4 : 4*size;
break;
case 'D':
if (size>0) p->printf ("Size not yet implemented\n");
if (p->disasm && p->user)
i += p->disasm (p->user, seeki);
break;
case 'o':
r_print_format_octal (p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i+= (size==-1) ? 4 : 4*size;
break;
case 'x':
r_print_format_hexflag(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i+= (size==-1) ? 4 : 4*size;
break;
case 'w':
r_print_format_word(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
i+= (size==-1) ? 2 : 2*size;
break;
case 'z': // zero terminated string
if (MUSTSET) {
int buflen = strlen ((const char *)buf);
if (buflen>seeki) {
buflen = strlen ((const char *)buf+seeki);
}
if (strlen (setval) > buflen) {
eprintf ("Warning: new string is longer than previous one \n");
}
realprintf ("w %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
if (json)
p->printf ("%d,\"string\":\"", seeki);
else
p->printf ("0x%08"PFMT64x" = ", seeki);
for (; ((size || size==-1) && buf[i]) && i<len; i++) {
if (IS_PRINTABLE (buf[i]))
p->printf ("%c", buf[i]);
else p->printf (".");
size -= (size==-1) ? 0 : 1;
}
if (json) p->printf ("\"}");
}
if (size == -1)
i++;
else
while (size--) i++;
break;
case 'Z': // zero terminated wide string
if (MUSTSET) {
if ((size = strlen(setval)) > r_wstr_clen((char*)(buf+seeki)))
eprintf ("Warning: new string is longer than previous one\n");
realprintf ("ww %s @ 0x%08"PFMT64x"\n", setval, seeki);
} else {
p->printf ("0x%08"PFMT64x" = ", seeki);
for (; ((size || size==-1) && buf[i]) && i<len; i+=2) {
if (IS_PRINTABLE (buf[i]))
p->printf ("%c", buf[i]);
else p->printf (".");
size -= (size==-1) ? 0 : 1;
}
}
if (size == -1)
i+=2;
else
while (size--) i+=2;
break;
case 's':
if (r_print_format_string (p, seeki, addr64, addr, 0, json) == 0)
i += (size==-1) ? 4 : 4*size;
break;
case 'S':
if (r_print_format_string (p, seeki, addr64, addr, 1, json) == 0)
i += (size==-1) ? 8 : 8*size;
break;
case 'B': // resolve bitfield
{
char *structname, *osn;
char *bitfield = NULL;
structname = osn = strdup (r_str_word_get0 (args, idx-1));
switch (size) {
case 1: addr &= UT8_MAX; break;
case 2: addr &= UT16_MAX; break;
case 4: addr &= UT32_MAX; break;
}
if (*structname == '(') {
name = strchr (structname, ')');
} else {
eprintf ("Bitfield name missing (%s)\n", structname);
free (structname);
goto beach;
}
structname++;
if (name) *(name++) = '\0';
else eprintf ("No ')'\n");
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (p->get_bitfield)
bitfield = p->get_bitfield (p->user, structname, addr);
if (bitfield && *bitfield) {
if (json) p->printf ("\"%s\"}", bitfield);
else p->printf (" %s (bitfield) = %s\n", name, bitfield);
} else {
if (json) p->printf ("\"`tb %s 0x%x`\"}", structname, addr);
else p->printf (" %s (bitfield) = `tb %s 0x%x`\n",
name, structname, addr);
}
i+=(size==-1)?1:size;
free (osn);
free (bitfield);
}
break;
case 'E': // resolve enum
{
char *enumname, *osn;
char *enumvalue = NULL;
enumname = osn = strdup (r_str_word_get0 (args, idx-1));
switch (size) {
case 1: addr &= UT8_MAX; break;
case 2: addr &= UT16_MAX; break;
case 4: addr &= UT32_MAX; break;
}
if (*enumname == '(') {
name = strchr (enumname, ')');
} else {
eprintf ("Enum name missing (%s)\n", enumname);
free (enumname);
goto beach;
}
enumname++;
if (name) *(name++) = '\0';
else eprintf ("No ')'\n");
if (!json)
p->printf ("0x%08"PFMT64x" = ", seeki);
if (p->get_enumname)
enumvalue = p->get_enumname (p->user, enumname, addr);
if (enumvalue && *enumvalue) {
if (json) p->printf ("\"%s\"}", enumname);
else p->printf (" %s (enum) = 0x%"PFMT64x" ; %s\n",
name, addr, enumvalue);
} else {
if (json) p->printf ("\"`te %s 0x%x`\"}", enumname, addr);
else p->printf (" %s (enum) = `te %s 0x%x`\n",
name, enumname, addr);
}
i+=(size==-1)?1:size;
free (osn);
free (enumvalue);
}
break;
case '?':
{
int s = 0;
char *structname, *osn;
structname = osn = strdup (r_str_word_get0 (args, idx-1));
if (*structname == '(') {
name = strchr (structname, ')');
} else {
eprintf ("Struct name missing (%s)\n", structname);
free (structname);
goto beach;
}
structname++;
if (name) *(name++) = '\0';
else eprintf ("No ')'\n");
if (!json) {
p->printf ("struct<%s>\n", structname);
if (isptr) {
p->printf (namefmt, "----");
p->printf ("\n");
}
} else {
if (isptr)
p->printf ("%d},", seeki);
else
p->printf ("[");
}
if (flag) slide+=STRUCTFLAG;
oldslide = slide;
slide += (isptr) ? STRUCTPTR : NESTEDSTRUCT;
if (size == -1) {
s = r_print_format_struct (p, seeki,
buf+i, len, structname, slide, json);
i+= (isptr) ? 4 : s;
} else {
p->printf ("[\n");
s = r_print_format_struct (p, seeki,
buf+i, len, structname, slide, json);
i+= (isptr) ? 4 : s;
size--;
while (size--) {
p->printf (",\n");
s = r_print_format_struct (p, seeki,
buf+i, len, structname, slide, json);
i+= (isptr) ? 4 : s;
}
if (json) p->printf ("]]}");
else p->printf ("]");
}
oldslide = slide;
slide -= (isptr) ? STRUCTPTR : NESTEDSTRUCT;
if (flag) {
oldslide = slide;
slide-=STRUCTFLAG;
}
free (osn);
break;
}
default:
/* ignore unknown chars */
invalid = 1;
break;
}
if (!json && !flag && (!MUSTSEE || MUSTSET))
idx++;
if (viewflags && p->offname) {
const char *s = p->offname (p->user, seeki);
if (s)
p->printf ("@(%s)", s);
s = p->offname (p->user, addr);
if (s)
p->printf ("*(%s)", s);
}
if (tmp != 'D' && !invalid && name==NULL)
p->printf ("\n");
last = tmp;
}
if (otimes>1) {
if (json) p->printf ("]");
else p->printf ("}\n");
}
arg = orig;
oldslide = 0;
// if (json && arg != argend && slide>=oldslide) p->printf (",\n");
}
if (json && slide==0) p->printf("]");
if (oldprintf)
p->printf = oldprintf;
beach:
free (buf);
if (args)
free (args);
return i;
}