2014-09-03 10:25:29 +02:00
|
|
|
/* radare - LGPL - Copyright 2007-2014 - pancake */
|
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"
|
|
|
|
|
2014-09-16 00:12:59 +02:00
|
|
|
#define NOPTR 0
|
|
|
|
#define PTRSEEK 1
|
|
|
|
#define PTRBACK 2
|
|
|
|
#define NULLPTR 3
|
|
|
|
#define STRUCTPTR 100
|
|
|
|
#define NESTEDSTRUCT 1
|
|
|
|
#define STRUCTFLAG 10000
|
|
|
|
#define NESTDEPTH 14
|
|
|
|
|
2014-08-01 19:33:52 +02:00
|
|
|
static int (*realprintf)(const char *str, ...);
|
2013-02-01 03:15:48 +01:00
|
|
|
static int nullprintf(const char *fmt, ...) { return 0; }
|
|
|
|
|
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 (
|
2014-07-17 00:22:43 +02:00
|
|
|
"Usage: pf[.key[.field[=value]]|[ val]]|[times][ [size] format] [arg0 arg1 ...]\n"
|
2013-01-22 05:06:12 +01:00
|
|
|
"Examples:\n"
|
2013-02-01 03:15:48 +01:00
|
|
|
" pf 10xiz pointer length string\n"
|
|
|
|
" pf {array_size}b @ array_base\n"
|
2014-10-21 11:49:33 +02:00
|
|
|
" pf [4]w[7]i # like pf w..i...\n"
|
|
|
|
" pfo # list all format files\n"
|
|
|
|
" pfo trx.pf # load that format definition file\n"
|
2013-02-01 03:15:48 +01:00
|
|
|
" pf. # list all formats\n"
|
|
|
|
" pf.obj xxdz prev next size name\n"
|
2014-10-21 11:49:33 +02:00
|
|
|
" pf obj=xxdz prev next size name # same as above\n"
|
2013-02-01 03:15:48 +01:00
|
|
|
" pf.obj # run stored format\n"
|
|
|
|
" pf.obj.name # show string inside object\n"
|
|
|
|
" pf.obj.size=33 # set new size\n"
|
2013-01-22 05:06:12 +01:00
|
|
|
"Format chars:\n"
|
2009-03-11 11:42:11 +00:00
|
|
|
" e - temporally swap endian\n"
|
|
|
|
//" D - double (8 bytes)\n"
|
2013-02-01 03:15:48 +01:00
|
|
|
" f - float value (4 bytes)\n"
|
|
|
|
" b - byte (unsigned)\n"
|
2014-10-14 17:49:00 +02:00
|
|
|
" B - resolve enum bitfield (see t?) `pf B (Bitfield_type)arg_name`\n" // B must be for binary ??
|
2014-09-21 01:39:18 +02:00
|
|
|
" c - char (signed byte)\n"
|
2014-10-14 17:49:00 +02:00
|
|
|
" E - resolve enum name (see t?) `pf E (Enum_type)arg_name`\n"
|
2014-07-17 00:22:43 +02:00
|
|
|
" X - show n hexpairs (default n=1)"
|
2010-06-25 11:22:14 +02:00
|
|
|
" i - %%i integer value (4 bytes)\n"
|
2013-02-01 03:15:48 +01:00
|
|
|
" w - word (2 bytes unsigned short in hex)\n"
|
2009-03-11 11:42:11 +00:00
|
|
|
" q - quadword (8 bytes)\n"
|
2013-02-01 03:15:48 +01:00
|
|
|
" p - pointer reference (2, 4 or 8 bytes)\n"
|
2014-09-21 01:39:18 +02:00
|
|
|
" T - show Ten first bytes of buffer\n" // B must be for binary ??
|
2013-02-01 03:15:48 +01:00
|
|
|
" d - 0x%%08x hexadecimal value (4 bytes)\n"
|
2013-02-25 09:36:07 +01:00
|
|
|
" D - disassemble one opcode\n"
|
2014-09-08 12:20:50 +02:00
|
|
|
" o - 0x%%08o octal value (4 byte)\n"
|
2011-06-09 01:20:02 +02:00
|
|
|
" x - 0x%%08x hexadecimal value and flag (fd @ addr)\n"
|
2014-09-21 01:39:18 +02:00
|
|
|
" X - show formatted hexpairs\n" // B must be for binary ??
|
2009-03-11 11:42:11 +00:00
|
|
|
" z - \\0 terminated string\n"
|
|
|
|
" Z - \\0 terminated wide string\n"
|
2013-02-01 03:15:48 +01:00
|
|
|
" s - 32bit pointer to string (4 bytes)\n"
|
|
|
|
" S - 64bit pointer to string (8 bytes)\n"
|
2013-03-08 01:03:37 +01:00
|
|
|
//" t - unix timestamp string\n"
|
2014-09-27 22:43:34 +02:00
|
|
|
" ? - data structure `pf ? (struct_type)struct_name`\n"
|
2013-08-11 14:23:51 +02:00
|
|
|
" * - next char is pointer (honors asm.bits)\n"
|
2011-05-20 20:47:40 +02:00
|
|
|
" + - toggle show flags for each offset\n"
|
2013-10-09 00:29:49 +02:00
|
|
|
" : - skip 4 bytes\n"
|
2009-03-11 11:42:11 +00:00
|
|
|
" . - skip 1 byte\n");
|
|
|
|
}
|
2014-09-03 10:25:29 +02:00
|
|
|
|
2014-08-01 11:46:08 +02:00
|
|
|
static void updateAddr(const ut8 *buf, int i, int endian, ut64 *addr, ut64 *addr64) {
|
2014-09-17 12:08:27 +02:00
|
|
|
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)));
|
|
|
|
}
|
2014-08-01 00:15:17 +02:00
|
|
|
}
|
2009-03-11 11:42:11 +00:00
|
|
|
|
2014-09-17 12:08:27 +02:00
|
|
|
static void r_print_format_quadword(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-09-17 12:08:27 +02:00
|
|
|
ut64 addr64;
|
|
|
|
updateAddr (buf, i, endian, NULL, &addr64);
|
2014-08-01 19:33:52 +02:00
|
|
|
if (mustset) {
|
|
|
|
realprintf ("wv8 %s @ 0x%08"PFMT64x"\n", setval, seeki);
|
|
|
|
} else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json) {
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
|
|
|
p->printf ("(qword) ");
|
|
|
|
}
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size==-1)
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("%d", addr64);
|
|
|
|
else
|
|
|
|
p->printf ("0x%016"PFMT64x, addr64);
|
2014-09-17 12:08:27 +02:00
|
|
|
else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("[ %d", addr64);
|
|
|
|
else
|
|
|
|
p->printf ("[ 0x%016"PFMT64x, addr64);
|
2014-09-17 12:08:27 +02:00
|
|
|
size--;
|
|
|
|
i+=8;
|
|
|
|
while (size--) {
|
|
|
|
updateAddr (buf, i, endian, NULL, &addr64);
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf (", %d", addr64);
|
|
|
|
else
|
|
|
|
p->printf (", 0x%016"PFMT64x, addr64);
|
2014-09-17 12:08:27 +02:00
|
|
|
i+=8;
|
|
|
|
}
|
|
|
|
p->printf (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-08-01 19:33:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-17 12:08:27 +02:00
|
|
|
static void r_print_format_byte(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-08-01 19:33:52 +02:00
|
|
|
if (mustset) {
|
2014-09-16 00:12:59 +02:00
|
|
|
realprintf ("\"w %s\" @ 0x%08"PFMT64x"\n", setval, seeki);
|
2014-08-01 19:33:52 +02:00
|
|
|
} else {
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size==-1)
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("%d", buf[i]);
|
|
|
|
else
|
|
|
|
p->printf ("%d ; 0x%02x ; '%c'", buf[i], buf[i],
|
|
|
|
IS_PRINTABLE (buf[i])?buf[i]:0);
|
2014-09-17 12:08:27 +02:00
|
|
|
else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("[ %d", buf[i]);
|
|
|
|
else
|
|
|
|
p->printf ("[ %d ; 0x%02x ; '%c'", buf[i], buf[i],
|
|
|
|
IS_PRINTABLE (buf[i])?buf[i]:0);
|
2014-09-17 12:08:27 +02:00
|
|
|
size--;
|
|
|
|
i++;
|
|
|
|
while (size--) {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf (", %d", buf[i]);
|
|
|
|
else
|
|
|
|
p->printf (", %d ; 0x%02x ; '%c'", buf[i], buf[i],
|
|
|
|
IS_PRINTABLE (buf[i])?buf[i]:0);
|
2014-09-17 12:08:27 +02:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
p->printf (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-08-01 19:33:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-17 12:08:27 +02:00
|
|
|
static void r_print_format_char(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-08-01 19:33:52 +02:00
|
|
|
if (mustset) {
|
2014-09-16 00:12:59 +02:00
|
|
|
realprintf ("\"w %s\" @ 0x%08"PFMT64x"\n", setval, seeki);
|
2014-08-01 19:33:52 +02:00
|
|
|
} else {
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size==-1)
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("\"%c\"", buf[i]);
|
|
|
|
else
|
|
|
|
p->printf (" %d ; '%c'", buf[i], buf[i], buf[i],
|
|
|
|
IS_PRINTABLE (buf[i])?buf[i]:0);
|
2014-09-17 12:08:27 +02:00
|
|
|
else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("[ \"%c\"", buf[i]);
|
|
|
|
else
|
|
|
|
p->printf ("[ %d ; '%c'", buf[i], buf[i], buf[i],
|
|
|
|
IS_PRINTABLE (buf[i])?buf[i]:0);
|
2014-09-17 12:08:27 +02:00
|
|
|
size--;
|
|
|
|
i++;
|
|
|
|
while (size--) {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf (", \"%c\"", buf[i]);
|
|
|
|
else
|
|
|
|
p->printf (", %d ; '%c'", buf[i], buf[i], buf[i],
|
|
|
|
IS_PRINTABLE (buf[i])?buf[i]:0);
|
2014-09-17 12:08:27 +02:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
p->printf (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-08-01 19:33:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-26 02:22:58 +02:00
|
|
|
static int r_print_format_ptrstring(const RPrint* p, ut64 seeki, ut64 addr64, ut64 addr, int is64, int json) {
|
2014-08-01 19:33:52 +02:00
|
|
|
ut8 buffer[255];
|
|
|
|
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;
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("%d,\"string\":\"%s\"}", seeki, addr);
|
|
|
|
else {
|
|
|
|
p->printf ("0x%08"PFMT64x" -> 0x%08"PFMT64x" ", seeki, addr);
|
|
|
|
p->printf ("%s", buffer);
|
|
|
|
}
|
2014-08-01 19:33:52 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: support unsigned int?
|
2014-09-17 12:08:27 +02:00
|
|
|
static void r_print_format_hex(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-09-17 12:08:27 +02:00
|
|
|
ut64 addr;
|
|
|
|
updateAddr (buf, i, endian, &addr, NULL);
|
2014-08-01 19:33:52 +02:00
|
|
|
if (mustset) {
|
|
|
|
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
|
|
|
|
} else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json)
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size==-1)
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("%d", addr);
|
|
|
|
else
|
|
|
|
p->printf ("%"PFMT64d, addr);
|
2014-09-17 12:08:27 +02:00
|
|
|
else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("[ %d", addr);
|
|
|
|
else
|
|
|
|
p->printf ("[ %"PFMT64d, addr);
|
2014-09-17 12:08:27 +02:00
|
|
|
size--;
|
|
|
|
i+=4;
|
|
|
|
while (size--) {
|
|
|
|
updateAddr (buf, i, endian, &addr, NULL);
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf (", %d", addr);
|
|
|
|
else
|
|
|
|
p->printf (", %"PFMT64d, addr);
|
2014-09-17 12:08:27 +02:00
|
|
|
i+=4;
|
|
|
|
}
|
|
|
|
p->printf (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-08-01 19:33:52 +02:00
|
|
|
}
|
|
|
|
}
|
2014-09-08 12:20:50 +02:00
|
|
|
|
2014-09-17 12:08:27 +02:00
|
|
|
static void r_print_format_octal (const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-09-17 12:08:27 +02:00
|
|
|
ut64 addr;
|
|
|
|
updateAddr (buf, i, endian, &addr, NULL);
|
2014-09-08 12:20:50 +02:00
|
|
|
if (mustset) {
|
|
|
|
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
|
|
|
|
} else {
|
|
|
|
ut32 addr32 = (ut32)addr;
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json) {
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
|
|
|
p->printf ("(octal) ");
|
|
|
|
}
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size==-1)
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("%d", addr32);
|
|
|
|
else
|
|
|
|
p->printf ("0%08"PFMT64o, addr32);
|
2014-09-17 12:08:27 +02:00
|
|
|
else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("[ %d", addr32);
|
|
|
|
else
|
|
|
|
p->printf ("[ 0%08"PFMT64o, addr32);
|
2014-09-17 12:08:27 +02:00
|
|
|
size--;
|
|
|
|
i+=4;
|
|
|
|
while (size--) {
|
|
|
|
updateAddr (buf, i, endian, &addr, NULL);
|
|
|
|
addr32 = (ut32)addr;
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf (", %d", addr32);
|
|
|
|
else
|
|
|
|
p->printf (", 0%08"PFMT64o, addr32);
|
2014-09-17 12:08:27 +02:00
|
|
|
i+=4;
|
|
|
|
}
|
|
|
|
p->printf (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-09-08 12:20:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-17 12:08:27 +02:00
|
|
|
static void r_print_format_hexflag(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-09-17 12:08:27 +02:00
|
|
|
ut64 addr;
|
|
|
|
updateAddr (buf, i, endian, &addr, NULL);
|
2014-08-01 19:33:52 +02:00
|
|
|
if (mustset) {
|
|
|
|
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
|
|
|
|
} else {
|
|
|
|
ut32 addr32 = (ut32)addr;
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json)
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size==-1)
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("%d", addr32);
|
|
|
|
else
|
|
|
|
p->printf ("0x%08"PFMT64x, addr32);
|
2014-09-17 12:08:27 +02:00
|
|
|
else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("[ %d", addr32);
|
|
|
|
else
|
|
|
|
p->printf ("[ 0x%08"PFMT64x, addr32);
|
2014-09-17 12:08:27 +02:00
|
|
|
size--;
|
|
|
|
i+=4;
|
|
|
|
while (size--) {
|
|
|
|
updateAddr (buf, i, endian, &addr, NULL);
|
|
|
|
addr32 = (ut32)addr;
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf (",%d", addr32);
|
|
|
|
else
|
|
|
|
p->printf (", 0x%08"PFMT64x, addr32);
|
2014-09-17 12:08:27 +02:00
|
|
|
i+=4;
|
|
|
|
}
|
|
|
|
p->printf (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-08-01 19:33:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-26 02:22:58 +02:00
|
|
|
static int r_print_format_10bytes(const RPrint* p, int mustset, const char* setval, ut64 seeki, ut64 addr, ut8* buf, int json) {
|
2014-08-01 19:33:52 +02:00
|
|
|
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);
|
2014-10-26 02:22:58 +02:00
|
|
|
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;
|
|
|
|
}
|
2014-08-01 19:33:52 +02:00
|
|
|
for (j=0; j<10; j++)
|
|
|
|
if (IS_PRINTABLE (buf[j]))
|
|
|
|
p->printf ("%c", buf[j]);
|
|
|
|
else
|
|
|
|
p->printf (".");
|
|
|
|
p->printf (")");
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-17 12:08:27 +02:00
|
|
|
static int r_print_format_hexpairs(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int size, int json) {
|
2014-08-02 02:19:43 +02:00
|
|
|
int j;
|
|
|
|
|
|
|
|
if (mustset) {
|
|
|
|
realprintf ("?e pf X not yet implemented\n");
|
|
|
|
} else {
|
|
|
|
size = (size < 1) ? 1 : size;
|
2014-10-26 02:22:58 +02:00
|
|
|
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;
|
|
|
|
}
|
2014-08-02 02:19:43 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-09-17 12:08:27 +02:00
|
|
|
static void r_print_format_float(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-09-17 12:08:27 +02:00
|
|
|
ut64 addr;
|
|
|
|
updateAddr (buf, i, endian, &addr, NULL);
|
|
|
|
if (mustset) {
|
2014-08-02 02:19:43 +02:00
|
|
|
realprintf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki);
|
2014-09-17 12:08:27 +02:00
|
|
|
} else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json)
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-09-17 12:08:27 +02:00
|
|
|
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 (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-09-17 12:08:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void r_print_format_word(const RPrint* p, int endian, int mustset,
|
2014-10-26 02:22:58 +02:00
|
|
|
const char* setval, ut64 seeki, ut8* buf, int i, int size, int json) {
|
2014-09-17 12:08:27 +02:00
|
|
|
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 {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json)
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size==-1)
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("%d", addr);
|
|
|
|
else
|
|
|
|
p->printf ("0x%04x", addr);
|
2014-09-17 12:08:27 +02:00
|
|
|
else {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("[ %d", addr);
|
|
|
|
else
|
|
|
|
p->printf ("[ 0x%04x", addr);
|
2014-09-17 12:08:27 +02:00
|
|
|
size--;
|
|
|
|
i+=2;
|
|
|
|
while (size--) {
|
|
|
|
if (endian)
|
|
|
|
addr = (*(buf+i))<<8 | (*(buf+i+1));
|
|
|
|
else addr = (*(buf+i+1))<<8 | (*(buf+i));
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf (", %d", addr);
|
|
|
|
else
|
|
|
|
p->printf (", 0x%04x", addr);
|
2014-09-17 12:08:27 +02:00
|
|
|
i+=2;
|
|
|
|
}
|
|
|
|
p->printf (" ]");
|
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json) p->printf ("}");
|
2014-09-17 12:08:27 +02:00
|
|
|
}
|
2014-08-02 02:19:43 +02:00
|
|
|
}
|
|
|
|
|
2014-09-03 10:25:29 +02:00
|
|
|
// XXX: this is very incomplete. must be updated to handle all format chars
|
2014-10-07 17:51:15 +02:00
|
|
|
static int computeStructSize(char *format, RPrint *p) {
|
|
|
|
char *end = strchr(format, ' '), *args, *fmt=format;
|
|
|
|
int size = 0, tabsize=0, i, idx=0;
|
2014-08-31 02:09:35 +02:00
|
|
|
if (!end)
|
|
|
|
return -1;
|
2014-08-07 23:59:43 +02:00
|
|
|
*end = 0;
|
2014-10-01 10:48:43 +02:00
|
|
|
args = strdup (end+1);
|
|
|
|
r_str_word_set0 (args);
|
2014-08-07 23:59:43 +02:00
|
|
|
for (i=0; i<strlen(fmt); i++) {
|
2014-10-07 17:51:15 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-09-03 10:25:29 +02:00
|
|
|
switch (fmt[i]) {
|
2014-09-06 15:58:46 +02:00
|
|
|
case 'c':
|
2014-10-07 17:51:15 +02:00
|
|
|
case '.':
|
|
|
|
size+=tabsize*1;
|
2014-08-07 23:59:43 +02:00
|
|
|
break;
|
|
|
|
case 'w':
|
2014-10-07 17:51:15 +02:00
|
|
|
size += tabsize*2;
|
2014-09-06 15:58:46 +02:00
|
|
|
break;
|
|
|
|
case 'd':
|
2014-10-07 17:51:15 +02:00
|
|
|
case 'i':
|
|
|
|
case 'x':
|
|
|
|
case 'f':
|
2014-09-06 15:58:46 +02:00
|
|
|
case 's':
|
|
|
|
case ':':
|
2014-10-07 17:51:15 +02:00
|
|
|
size += tabsize*4;
|
2014-09-06 15:58:46 +02:00
|
|
|
break;
|
2014-10-07 17:51:15 +02:00
|
|
|
case 'S':
|
|
|
|
case 'q':
|
|
|
|
size += tabsize*8;
|
2014-09-06 15:58:46 +02:00
|
|
|
break;
|
2014-08-07 23:59:43 +02:00
|
|
|
case '*':
|
2014-10-07 17:51:15 +02:00
|
|
|
size += tabsize*4;
|
2014-08-07 23:59:43 +02:00
|
|
|
i++;
|
2014-09-06 15:58:46 +02:00
|
|
|
break;
|
2014-10-13 15:27:56 +02:00
|
|
|
case 'B':
|
|
|
|
case 'E':
|
|
|
|
switch (tabsize) {
|
|
|
|
case 1: size+=1; break;
|
|
|
|
case 2: size+=2; break;
|
|
|
|
case 4: size+=4; break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
break;
|
2014-10-01 10:48:43 +02:00
|
|
|
case '?':
|
|
|
|
{
|
|
|
|
char *endname = NULL, *format = 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!=NULL) *endname = '\0';
|
|
|
|
format = strdup(r_strht_get (p->formats, structname+1));
|
2014-10-07 17:51:15 +02:00
|
|
|
size += tabsize*computeStructSize (format, p);
|
2014-10-01 10:48:43 +02:00
|
|
|
free (structname);
|
|
|
|
break;
|
|
|
|
}
|
2014-08-07 23:59:43 +02:00
|
|
|
// TODO continue list
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2014-10-01 10:48:43 +02:00
|
|
|
idx++;
|
2014-08-07 23:59:43 +02:00
|
|
|
}
|
2014-10-01 10:48:43 +02:00
|
|
|
free (args);
|
2014-10-07 17:51:15 +02:00
|
|
|
free (format);
|
2014-08-07 23:59:43 +02:00
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2014-10-26 02:22:58 +02:00
|
|
|
static int r_print_format_struct(RPrint* p, ut64 seek, const ut8* b, int len, char *name, int slide, int json) {
|
2014-08-07 23:59:43 +02:00
|
|
|
const char *fmt;
|
2014-09-16 00:12:59 +02:00
|
|
|
int flag = (slide>=STRUCTFLAG)?SEEFLAG:-1;
|
2014-10-26 02:22:58 +02:00
|
|
|
flag = (json)?JSONOUTPUT:flag;
|
2014-09-16 00:12:59 +02:00
|
|
|
if ((slide%STRUCTPTR) > NESTDEPTH) {
|
2014-09-03 10:25:29 +02:00
|
|
|
eprintf ("Too much nested struct, recursion too deep...\n");
|
2014-08-07 23:59:43 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2014-09-14 15:17:45 +02:00
|
|
|
if (flag) p->printf = realprintf;
|
2014-08-07 23:59:43 +02:00
|
|
|
fmt = r_strht_get (p->formats, name);
|
2014-09-20 22:58:46 +02:00
|
|
|
if (!fmt || !*fmt) {
|
2014-09-23 15:24:56 +02:00
|
|
|
eprintf ("Undefined struct '%s'.\n", name);
|
2014-09-20 22:58:46 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2014-09-14 15:17:45 +02:00
|
|
|
r_print_format (p, seek, b, len, fmt, flag, NULL);
|
2014-10-01 10:48:43 +02:00
|
|
|
return computeStructSize(strdup(fmt), p);
|
2014-08-07 23:59:43 +02:00
|
|
|
}
|
|
|
|
|
2014-09-09 21:08:47 +02:00
|
|
|
R_API int r_print_format(RPrint *p, ut64 seek, const ut8* b, const int len,
|
|
|
|
const char *fmt, int elem, const char *setval) {
|
2014-07-14 15:05:45 +02:00
|
|
|
int nargs, i, j, invalid, nexti, idx, times, otimes, endian, isptr = 0;
|
2013-02-01 03:15:48 +01:00
|
|
|
int (*oldprintf)(const char *str, ...);
|
|
|
|
const char *argend = fmt+strlen (fmt);
|
|
|
|
ut64 addr = 0, addr64 = 0, seeki = 0;;
|
2014-04-28 11:37:48 +02:00
|
|
|
char *args = NULL, *bracket, tmp, last = 0;
|
2009-02-05 22:08:46 +01:00
|
|
|
const char *arg = fmt;
|
2014-10-26 02:22:58 +02:00
|
|
|
int viewflags = 0, flag = (elem==SEEFLAG)?1:0, json = (elem==JSONOUTPUT)?1:0;
|
2013-02-01 03:15:48 +01:00
|
|
|
char namefmt[8];
|
2014-10-26 02:22:58 +02:00
|
|
|
static int slide=0, oldslide=0;
|
2014-08-01 19:33:52 +02:00
|
|
|
ut8 *buf;
|
2014-10-26 02:22:58 +02:00
|
|
|
|
2014-08-02 14:47:45 +02:00
|
|
|
nexti = nargs = i = j = 0;
|
2013-02-01 03:15:48 +01:00
|
|
|
|
2014-08-02 14:47:45 +02:00
|
|
|
if (len < 1)
|
|
|
|
return 0;
|
2013-08-11 14:23:51 +02:00
|
|
|
buf = malloc (len);
|
2014-08-02 14:47:45 +02:00
|
|
|
if (!buf)
|
|
|
|
return 0;
|
2013-08-11 14:23:51 +02:00
|
|
|
memcpy (buf, b, len);
|
2013-02-22 21:05:08 +01:00
|
|
|
endian = p->big_endian;
|
|
|
|
|
2013-02-01 03:15:48 +01:00
|
|
|
oldprintf = NULL;
|
|
|
|
realprintf = p->printf;
|
2009-02-05 22:08:46 +01:00
|
|
|
|
2011-05-20 19:49:50 +02:00
|
|
|
while (*arg && iswhitechar (*arg)) arg++;
|
2014-08-02 14:47:45 +02:00
|
|
|
|
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++;
|
2014-08-02 02:19:43 +02:00
|
|
|
|
2010-03-01 10:49:04 +01:00
|
|
|
bracket = strchr (arg,'{');
|
2009-02-05 22:08:46 +01:00
|
|
|
if (bracket) {
|
2014-09-08 12:20:50 +02: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");
|
2013-08-11 14:23:51 +02:00
|
|
|
goto beach;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
*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
|
|
|
}
|
|
|
|
|
2014-09-06 04:22:42 +02:00
|
|
|
if (*arg=='\0' || *arg=='?') {
|
2011-03-04 13:08:40 +01:00
|
|
|
print_format_help (p);
|
2013-08-11 14:23:51 +02:00
|
|
|
goto beach;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-08-02 02:19:43 +02:00
|
|
|
|
2009-02-05 22:08:46 +01:00
|
|
|
/* 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);
|
2014-07-27 20:06:09 +02:00
|
|
|
nargs = r_str_word_set0 (args);
|
2009-02-05 22:08:46 +01:00
|
|
|
if (nargs == 0)
|
2010-03-01 10:49:04 +01:00
|
|
|
R_FREE (args);
|
2011-09-06 00:38:56 +02:00
|
|
|
for (i=0; i<nargs; i++) {
|
2014-08-02 02:19:43 +02:00
|
|
|
const int len = strlen (r_str_word_get0 (args, i));
|
|
|
|
if (len > maxl)
|
|
|
|
maxl = len;
|
2009-03-12 00:42:32 +00:00
|
|
|
}
|
|
|
|
l++;
|
2014-09-16 00:12:59 +02:00
|
|
|
snprintf (namefmt, sizeof (namefmt), "%%%ds : ", maxl+6*slide%STRUCTPTR);
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
#define ISPOINTED ((slide%STRUCTFLAG)/STRUCTPTR<=(oldslide%STRUCTFLAG)/STRUCTPTR)
|
|
|
|
#define ISNESTED ((slide%STRUCTPTR)<=(oldslide%STRUCTPTR))
|
2014-10-26 02:03:11 +01:00
|
|
|
if (json && slide==0) p->printf("[");
|
2014-10-26 02:22:58 +02:00
|
|
|
|
2009-02-05 22:08:46 +01:00
|
|
|
/* go format */
|
|
|
|
i = 0;
|
2014-08-02 02:19:43 +02:00
|
|
|
if (!times)
|
|
|
|
otimes = times = 1;
|
2011-09-06 00:38:56 +02:00
|
|
|
for (; times; times--) { // repeat N times
|
2009-02-05 22:08:46 +01:00
|
|
|
const char * orig = arg;
|
2014-10-26 02:22:58 +02:00
|
|
|
int first = 1;
|
|
|
|
if (otimes>1) {
|
|
|
|
if (json) {
|
2014-10-26 02:03:11 +01:00
|
|
|
if (otimes > times) p->printf (",");
|
|
|
|
p->printf ("[{\"index\":%d,\"offset\":%d},", otimes-times, seek+i);
|
2014-10-26 02:22:58 +02:00
|
|
|
} else
|
|
|
|
p->printf ("0x%08"PFMT64x" [%d] {\n", seek+i, otimes-times);
|
|
|
|
}
|
2011-09-06 00:38:56 +02:00
|
|
|
arg = orig;
|
2014-07-30 18:01:33 +02:00
|
|
|
for (idx=0; i<len && arg<argend && *arg; arg++) {
|
2014-07-16 23:11:33 +02:00
|
|
|
int size;
|
2014-08-07 23:59:43 +02:00
|
|
|
char *name = NULL;
|
2011-05-20 19:49:50 +02:00
|
|
|
seeki = seek+i;
|
2009-02-05 22:08:46 +01:00
|
|
|
addr = 0LL;
|
2014-07-14 15:05:45 +02:00
|
|
|
invalid = 0;
|
2014-07-16 23:11:33 +02:00
|
|
|
if (arg[0] == '[') {
|
|
|
|
char *end = strchr (arg,']');
|
|
|
|
if (end == NULL) {
|
|
|
|
eprintf ("No end bracket.\n");
|
|
|
|
goto beach;
|
|
|
|
}
|
2014-09-08 12:20:50 +02:00
|
|
|
*end = '\0';
|
2014-07-16 23:11:33 +02:00
|
|
|
size = r_num_math (NULL, arg+1);
|
|
|
|
arg = end + 1;
|
2014-09-08 12:20:50 +02:00
|
|
|
*end = ']';
|
2014-07-16 23:11:33 +02:00
|
|
|
} else {
|
|
|
|
size = -1;
|
|
|
|
}
|
2014-08-01 00:15:17 +02:00
|
|
|
updateAddr (buf, i, endian, &addr, &addr64);
|
|
|
|
|
2009-02-05 22:08:46 +01:00
|
|
|
tmp = *arg;
|
2014-07-30 18:01:33 +02:00
|
|
|
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json && otimes>1)
|
2014-07-27 20:04:27 +02:00
|
|
|
p->printf (" ");
|
|
|
|
#define MUSTSET (setval && elem == idx)
|
|
|
|
#define MUSTSEE (elem == -1 || elem == idx)
|
2014-10-26 02:22:58 +02:00
|
|
|
#define ISSTRUCT (tmp == '?' || (tmp == '*' && *(arg+1) == '?'))
|
|
|
|
if ((MUSTSEE && !flag)) {
|
2014-07-27 20:04:27 +02:00
|
|
|
if (!(MUSTSET)) {
|
|
|
|
if (oldprintf)
|
|
|
|
p->printf = oldprintf;
|
2014-07-30 18:01:33 +02:00
|
|
|
if (idx<nargs && tmp != 'e' && isptr == 0) {
|
2014-10-26 13:50:58 +01:00
|
|
|
const char *name = r_str_word_get0 (args, idx);
|
2014-10-26 02:22:58 +02:00
|
|
|
if (ISSTRUCT) {
|
|
|
|
if (*name == '(') {
|
|
|
|
name = strchr (name, ')')+1;
|
|
|
|
} else {
|
|
|
|
eprintf ("Struct name missing (%s)\n", name);
|
|
|
|
goto beach;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
p->printf (namefmt, name);
|
2014-07-30 18:01:33 +02:00
|
|
|
idx++;
|
|
|
|
}
|
2014-07-27 20:04:27 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!oldprintf)
|
|
|
|
oldprintf = p->printf;
|
|
|
|
p->printf = nullprintf;
|
|
|
|
}
|
|
|
|
|
2009-02-05 22:08:46 +01:00
|
|
|
feed_me_again:
|
2013-08-11 14:23:51 +02:00
|
|
|
switch (isptr) {
|
|
|
|
case 1:
|
2014-08-07 23:59:43 +02:00
|
|
|
{
|
2013-08-11 14:23:51 +02:00
|
|
|
nexti = i + (p->bits/8);
|
|
|
|
i = 0;
|
2014-08-07 23:59:43 +02:00
|
|
|
if(tmp == '?' )seeki = addr;
|
2013-08-11 14:23:51 +02:00
|
|
|
memset (buf, '\0', len);
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json)
|
|
|
|
p->printf ("(*0x%"PFMT64x") ", addr);
|
2014-09-16 00:12:59 +02:00
|
|
|
if (addr == 0) isptr = NULLPTR;
|
|
|
|
else isptr = PTRBACK;
|
2014-08-07 23:59:43 +02:00
|
|
|
if (/*addr<(b+len) && addr>=b && */p->iob.read_at) { /* The test was here to avoid segfault in the next line,
|
2014-09-14 15:17:45 +02:00
|
|
|
but len make it doesnt work... */
|
2013-08-11 14:23:51 +02:00
|
|
|
p->iob.read_at (p->iob.io, (ut64)addr, buf, len-4);
|
2014-08-01 00:15:17 +02:00
|
|
|
updateAddr (buf, i, endian, &addr, &addr64);
|
2013-08-11 14:23:51 +02:00
|
|
|
} else {
|
2014-08-31 16:10:38 +04:00
|
|
|
eprintf ("(SEGFAULT: cannot read memory at 0x%08"PFMT64x", Block: %s, blocksize: 0x%x)\n",
|
2014-08-07 23:59:43 +02:00
|
|
|
addr, b, len);
|
|
|
|
p->printf("\n");
|
|
|
|
goto beach;
|
|
|
|
}
|
2013-08-11 14:23:51 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
// restore state after pointer seek
|
|
|
|
i = nexti;
|
|
|
|
seeki = seek+i;
|
|
|
|
memcpy (buf, b, len);
|
2014-09-16 00:12:59 +02:00
|
|
|
isptr = NOPTR;
|
2013-08-11 14:23:51 +02:00
|
|
|
arg--;
|
|
|
|
continue;
|
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
if (tmp == 0 && last != '*')
|
|
|
|
break;
|
2014-08-02 02:19:43 +02:00
|
|
|
|
2009-02-05 22:08:46 +01:00
|
|
|
/* skip chars */
|
2011-05-20 19:49:50 +02:00
|
|
|
switch (tmp) {
|
2014-08-02 14:47:45 +02:00
|
|
|
case '*': // next char is a pointer
|
2014-09-16 00:12:59 +02:00
|
|
|
isptr = PTRSEEK;
|
2013-08-11 14:23:51 +02:00
|
|
|
arg++;
|
|
|
|
tmp = *arg; //last;
|
2009-02-05 22:08:46 +01:00
|
|
|
goto feed_me_again;
|
2014-08-02 02:19:43 +02:00
|
|
|
case '+': // toggle view flags
|
2011-05-20 20:47:40 +02:00
|
|
|
viewflags = !viewflags;
|
|
|
|
continue;
|
2009-02-05 22:08:46 +01:00
|
|
|
case 'e': // tmp swap endian
|
2011-05-20 19:49:50 +02:00
|
|
|
endian ^= 1;
|
2009-02-05 22:08:46 +01:00
|
|
|
continue;
|
2014-08-02 02:19:43 +02:00
|
|
|
case ':': // skip 4 bytes
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size == -1) i+=4;
|
|
|
|
else
|
|
|
|
while (size--) i+=4;
|
2013-10-09 00:29:49 +02:00
|
|
|
continue;
|
2014-08-02 02:19:43 +02:00
|
|
|
case '.': // skip 1 byte
|
2014-09-17 12:08:27 +02:00
|
|
|
if (size == -1) i++;
|
|
|
|
else
|
|
|
|
while (size--) i++;
|
2009-02-05 22:08:46 +01:00
|
|
|
continue;
|
2014-08-02 02:19:43 +02:00
|
|
|
case 'p': // pointer reference
|
|
|
|
tmp = (p->bits == 64)? 'q': 'x';
|
2013-08-13 03:23:39 +02:00
|
|
|
//tmp = (sizeof (void*)==8)? 'q': 'x';
|
2011-07-12 17:26:56 +02:00
|
|
|
break;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-09-16 00:12:59 +02:00
|
|
|
if (flag && isptr != NULLPTR) {
|
2014-09-14 15:17:45 +02:00
|
|
|
if (tmp == '?') {
|
|
|
|
char *n = strdup (r_str_word_get0 (args, idx)+1);
|
|
|
|
char *par = strchr (n, ')');
|
2014-09-16 00:12:59 +02:00
|
|
|
if (par == NULL) {
|
|
|
|
eprintf ("No end parenthesis for struct name");
|
2014-09-23 08:48:58 +02:00
|
|
|
free (n);
|
2014-09-16 00:12:59 +02:00
|
|
|
goto beach;
|
|
|
|
} else {
|
2014-09-14 15:17:45 +02:00
|
|
|
*par = '.';
|
|
|
|
}
|
2014-09-16 00:12:59 +02:00
|
|
|
realprintf ("f %s_", n);
|
|
|
|
free(n);
|
2014-10-26 02:41:28 +02:00
|
|
|
} 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);
|
2014-09-14 15:17:45 +02:00
|
|
|
} 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++;
|
2014-09-09 21:08:47 +02:00
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
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)
|
2014-10-26 02:03:11 +01:00
|
|
|
p->printf (",");
|
2014-10-26 02:22:58 +02:00
|
|
|
else
|
|
|
|
first = 0;
|
|
|
|
} else if(oldslide!=0) {
|
2014-10-26 02:03:11 +01:00
|
|
|
p->printf ("]},");
|
2014-10-26 02:22:58 +02:00
|
|
|
}
|
|
|
|
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);
|
|
|
|
}
|
2014-07-27 20:04:27 +02:00
|
|
|
|
2014-09-16 00:12:59 +02:00
|
|
|
if (isptr == NULLPTR) {
|
2014-10-26 02:22:58 +02:00
|
|
|
if (json)
|
|
|
|
p->printf ("\"NULL\"}", tmp, seek+i);
|
|
|
|
else
|
|
|
|
p->printf ("NULL");
|
2014-09-16 00:12:59 +02:00
|
|
|
isptr = PTRBACK;
|
2014-08-07 23:59:43 +02:00
|
|
|
} else
|
2009-02-05 22:08:46 +01:00
|
|
|
/* cmt chars */
|
2011-03-04 13:08:40 +01:00
|
|
|
switch (tmp) {
|
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
|
2014-08-02 02:19:43 +02:00
|
|
|
case 'e': //WTF is this? 'e' is supposed to swap endians?!
|
|
|
|
if (size > 0)
|
|
|
|
p->printf ("Size not yet implemented\n");
|
2013-02-01 03:15:48 +01:00
|
|
|
if (MUSTSET) {
|
|
|
|
realprintf ("?e pf e not yet supported\n");
|
|
|
|
} else {
|
|
|
|
double doub;
|
|
|
|
memcpy (&doub, buf+i, sizeof (double));
|
|
|
|
p->printf ("0x%08"PFMT64x" = (double) ", seeki);
|
|
|
|
p->printf ("%e", doub);
|
|
|
|
i += 8;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'q':
|
2014-10-26 02:22:58 +02:00
|
|
|
r_print_format_quadword(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
|
2014-09-14 15:17:45 +02:00
|
|
|
i += (size==-1) ? 8 : 8*size;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
|
|
|
case 'b':
|
2014-10-26 02:22:58 +02:00
|
|
|
r_print_format_byte(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
|
2014-07-16 23:11:33 +02:00
|
|
|
i+= (size==-1) ? 1 : size;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
2012-06-22 03:49:25 +02:00
|
|
|
case 'c':
|
2014-09-17 12:08:27 +02:00
|
|
|
r_print_format_char (p, endian, MUSTSET,
|
2014-10-26 02:22:58 +02:00
|
|
|
setval, seeki, buf, i, size, json);
|
2014-07-16 23:11:33 +02:00
|
|
|
i+= (size==-1) ? 1 : size;
|
2012-06-22 03:49:25 +02:00
|
|
|
break;
|
2014-07-17 00:22:43 +02:00
|
|
|
case 'X':
|
2014-09-21 01:39:18 +02:00
|
|
|
size = r_print_format_hexpairs (p, endian, MUSTSET,
|
2014-10-26 02:22:58 +02:00
|
|
|
setval, seeki, buf, size, json);
|
2014-09-16 00:12:59 +02:00
|
|
|
i += size;
|
2014-07-17 00:22:43 +02:00
|
|
|
break;
|
2014-09-21 01:39:18 +02:00
|
|
|
case 'T':
|
2014-09-16 00:12:59 +02:00
|
|
|
if(r_print_format_10bytes(p, MUSTSET,
|
2014-10-26 02:22:58 +02:00
|
|
|
setval, seeki, addr, buf, json) == 0)
|
2014-09-16 00:12:59 +02:00
|
|
|
i += (size==-1) ? 4 : 4*size;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
2013-10-09 00:29:49 +02:00
|
|
|
case 'f':
|
2014-10-26 02:22:58 +02:00
|
|
|
r_print_format_float(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
|
2014-09-16 00:12:59 +02:00
|
|
|
i += (size==-1) ? 4 : 4*size;
|
2013-10-09 00:29:49 +02:00
|
|
|
break;
|
2010-06-25 11:22:14 +02:00
|
|
|
case 'i':
|
2014-08-01 19:33:52 +02:00
|
|
|
case 'd':
|
2014-10-26 02:22:58 +02:00
|
|
|
r_print_format_hex(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
|
2014-09-14 15:17:45 +02:00
|
|
|
i+= (size==-1) ? 4 : 4*size;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
2013-02-25 09:36:07 +01:00
|
|
|
case 'D':
|
2014-07-16 23:11:33 +02:00
|
|
|
if (size>0) p->printf ("Size not yet implemented\n");
|
2013-02-25 09:36:07 +01:00
|
|
|
if (p->disasm && p->user)
|
|
|
|
i += p->disasm (p->user, seeki);
|
|
|
|
break;
|
2014-09-08 12:20:50 +02:00
|
|
|
case 'o':
|
2014-10-26 02:22:58 +02:00
|
|
|
r_print_format_octal (p, endian, MUSTSET, setval, seeki, buf, i, size, json);
|
2014-09-14 15:17:45 +02:00
|
|
|
i+= (size==-1) ? 4 : 4*size;
|
2014-09-08 12:20:50 +02:00
|
|
|
break;
|
2013-02-01 03:15:48 +01:00
|
|
|
case 'x':
|
2014-10-26 02:22:58 +02:00
|
|
|
r_print_format_hexflag(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
|
2014-09-14 15:17:45 +02:00
|
|
|
i+= (size==-1) ? 4 : 4*size;
|
2013-02-01 03:15:48 +01:00
|
|
|
break;
|
2009-02-05 22:08:46 +01:00
|
|
|
case 'w':
|
2014-10-26 02:22:58 +02:00
|
|
|
r_print_format_word(p, endian, MUSTSET, setval, seeki, buf, i, size, json);
|
2014-09-14 15:17:45 +02:00
|
|
|
i+= (size==-1) ? 2 : 2*size;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
|
|
|
case 'z': // zero terminated string
|
2013-02-01 03:15:48 +01:00
|
|
|
if (MUSTSET) {
|
2014-09-14 15:17:45 +02:00
|
|
|
int buflen = strlen ((const char *)buf);
|
2014-09-13 00:36:50 +02:00
|
|
|
if (buflen>seeki) {
|
2014-09-14 15:17:45 +02:00
|
|
|
buflen = strlen ((const char *)buf+seeki);
|
2014-09-13 00:36:50 +02:00
|
|
|
}
|
|
|
|
if (strlen (setval) > buflen) {
|
2014-07-14 15:05:45 +02:00
|
|
|
eprintf ("Warning: new string is longer than previous one \n");
|
2014-09-13 00:36:50 +02:00
|
|
|
}
|
2014-09-14 15:17:45 +02:00
|
|
|
realprintf ("w %s @ 0x%08"PFMT64x"\n", setval, seeki);
|
2013-02-01 03:15:48 +01:00
|
|
|
} else {
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-07-16 23:11:33 +02:00
|
|
|
for (; ((size || size==-1) && buf[i]) && i<len; i++) {
|
2013-02-01 03:15:48 +01:00
|
|
|
if (IS_PRINTABLE (buf[i]))
|
|
|
|
p->printf ("%c", buf[i]);
|
|
|
|
else p->printf (".");
|
2014-07-16 23:11:33 +02:00
|
|
|
size -= (size==-1) ? 0 : 1;
|
2013-02-01 03:15:48 +01:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-07-16 23:11:33 +02:00
|
|
|
if (size == -1)
|
|
|
|
i++;
|
|
|
|
else
|
|
|
|
while (size--) i++;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
|
|
|
case 'Z': // zero terminated wide string
|
2014-07-14 15:05:45 +02:00
|
|
|
if (MUSTSET) {
|
2014-07-30 18:01:33 +02:00
|
|
|
if ((size = strlen(setval)) > r_wstr_clen((char*)(buf+seeki)))
|
2014-07-14 15:05:45 +02:00
|
|
|
eprintf ("Warning: new string is longer than previous one\n");
|
2014-09-14 15:17:45 +02:00
|
|
|
realprintf ("ww %s @ 0x%08"PFMT64x"\n", setval, seeki);
|
2014-07-14 15:05:45 +02:00
|
|
|
} else {
|
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
2014-07-16 23:29:02 +02:00
|
|
|
for (; ((size || size==-1) && buf[i]) && i<len; i+=2) {
|
2014-07-14 15:05:45 +02:00
|
|
|
if (IS_PRINTABLE (buf[i]))
|
|
|
|
p->printf ("%c", buf[i]);
|
|
|
|
else p->printf (".");
|
2014-09-14 15:17:45 +02:00
|
|
|
size -= (size==-1) ? 0 : 1;
|
2014-07-14 15:05:45 +02:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-07-16 23:29:02 +02:00
|
|
|
if (size == -1)
|
|
|
|
i+=2;
|
|
|
|
else
|
2014-09-14 15:17:45 +02:00
|
|
|
while (size--) i+=2;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
|
|
|
case 's':
|
2014-10-26 02:22:58 +02:00
|
|
|
if (r_print_format_ptrstring (p, seeki, addr64, addr, 0, json) == 0)
|
2014-09-14 15:17:45 +02:00
|
|
|
i += (size==-1) ? 4 : 4*size;
|
2009-02-05 22:08:46 +01:00
|
|
|
break;
|
2013-01-22 05:06:12 +01:00
|
|
|
case 'S':
|
2014-10-26 02:22:58 +02:00
|
|
|
if (r_print_format_ptrstring (p, seeki, addr64, addr, 1, json) == 0)
|
2014-09-14 15:17:45 +02:00
|
|
|
i += (size==-1) ? 8 : 8*size;
|
2013-01-22 05:06:12 +01:00
|
|
|
break;
|
2014-09-21 01:39:18 +02:00
|
|
|
case 'B': // resolve bitfield
|
|
|
|
{
|
|
|
|
char *structname, *osn;
|
|
|
|
char *bitfield = NULL;
|
|
|
|
structname = osn = strdup (r_str_word_get0 (args, idx-1));
|
2014-09-23 16:07:18 +02:00
|
|
|
switch (size) {
|
|
|
|
case 1: addr &= UT8_MAX; break;
|
|
|
|
case 2: addr &= UT16_MAX; break;
|
|
|
|
case 4: addr &= UT32_MAX; break;
|
|
|
|
}
|
2014-09-21 01:39:18 +02:00
|
|
|
if (*structname == '(') {
|
|
|
|
name = strchr (structname, ')');
|
|
|
|
} else {
|
2014-09-25 16:49:18 +02:00
|
|
|
eprintf ("Bitfield name missing (%s)\n", structname);
|
2014-09-21 01:39:18 +02:00
|
|
|
free (structname);
|
|
|
|
goto beach;
|
|
|
|
}
|
|
|
|
structname++;
|
|
|
|
if (name) *(name++) = '\0';
|
|
|
|
else eprintf ("No ')'\n");
|
2014-10-13 15:27:56 +02:00
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
|
|
|
if (p->get_bitfield)
|
|
|
|
bitfield = p->get_bitfield (p->user, structname, addr);
|
|
|
|
if (bitfield && *bitfield) {
|
|
|
|
p->printf (" %s (bitfield) = %s\n", name, bitfield);
|
2014-09-21 01:39:18 +02:00
|
|
|
} else {
|
2014-10-13 15:27:56 +02:00
|
|
|
p->printf (" %s (bitfield) = `tb %s 0x%x`\n",
|
|
|
|
name, structname, addr);
|
2014-09-21 01:39:18 +02:00
|
|
|
}
|
2014-10-13 15:27:56 +02:00
|
|
|
i+=(size==-1)?1:size;
|
|
|
|
|
2014-09-21 01:39:18 +02:00
|
|
|
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));
|
2014-09-23 16:07:18 +02:00
|
|
|
switch (size) {
|
|
|
|
case 1: addr &= UT8_MAX; break;
|
|
|
|
case 2: addr &= UT16_MAX; break;
|
|
|
|
case 4: addr &= UT32_MAX; break;
|
|
|
|
}
|
2014-09-21 01:39:18 +02:00
|
|
|
if (*enumname == '(') {
|
|
|
|
name = strchr (enumname, ')');
|
|
|
|
} else {
|
2014-09-25 16:49:18 +02:00
|
|
|
eprintf ("Enum name missing (%s)\n", enumname);
|
2014-09-21 01:39:18 +02:00
|
|
|
free (enumname);
|
|
|
|
goto beach;
|
|
|
|
}
|
|
|
|
enumname++;
|
|
|
|
if (name) *(name++) = '\0';
|
|
|
|
else eprintf ("No ')'\n");
|
2014-10-13 15:27:56 +02:00
|
|
|
p->printf ("0x%08"PFMT64x" = ", seeki);
|
|
|
|
if (p->get_enumname)
|
|
|
|
enumvalue = p->get_enumname (p->user, enumname, addr);
|
|
|
|
if (enumvalue && *enumvalue) {
|
|
|
|
p->printf (" %s (enum) = 0x%"PFMT64x" ; %s\n",
|
|
|
|
name, addr, enumvalue);
|
2014-09-21 01:39:18 +02:00
|
|
|
} else {
|
2014-10-13 15:27:56 +02:00
|
|
|
p->printf (" %s (enum) = `te %s 0x%x`\n",
|
|
|
|
name, enumname, addr);
|
2014-09-21 01:39:18 +02:00
|
|
|
}
|
2014-10-13 15:27:56 +02:00
|
|
|
i+=(size==-1)?1:size;
|
2014-10-07 17:51:15 +02:00
|
|
|
|
2014-09-21 01:39:18 +02:00
|
|
|
free (osn);
|
|
|
|
free (enumvalue);
|
|
|
|
}
|
|
|
|
break;
|
2014-08-07 23:59:43 +02:00
|
|
|
case '?':
|
|
|
|
{
|
2014-10-07 17:51:15 +02:00
|
|
|
int s = 0;
|
2014-09-21 01:39:18 +02:00
|
|
|
char *structname, *osn;
|
|
|
|
structname = osn = strdup (r_str_word_get0 (args, idx-1));
|
2014-09-14 15:17:45 +02:00
|
|
|
if (*structname == '(') {
|
|
|
|
name = strchr (structname, ')');
|
|
|
|
} else {
|
|
|
|
eprintf ("Struct name missing (%s)\n", structname);
|
2014-08-31 02:09:35 +02:00
|
|
|
free (structname);
|
2014-08-07 23:59:43 +02:00
|
|
|
goto beach;
|
|
|
|
}
|
2014-09-14 15:17:45 +02:00
|
|
|
structname++;
|
|
|
|
if (name) *(name++) = '\0';
|
|
|
|
else eprintf ("No ')'\n");
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json)
|
|
|
|
p->printf ("struct<%s>\n", structname);
|
|
|
|
else {
|
|
|
|
if (isptr)
|
2014-10-26 02:03:11 +01:00
|
|
|
p->printf ("%d},", seeki);
|
2014-10-26 02:22:58 +02:00
|
|
|
else
|
2014-10-26 02:03:11 +01:00
|
|
|
p->printf ("[");
|
2014-10-26 02:22:58 +02:00
|
|
|
}
|
2014-09-16 00:12:59 +02:00
|
|
|
if (flag) slide+=STRUCTFLAG;
|
2014-10-26 02:22:58 +02:00
|
|
|
oldslide = slide;
|
2014-09-16 00:12:59 +02:00
|
|
|
slide += (isptr) ? STRUCTPTR : NESTEDSTRUCT;
|
2014-10-07 17:51:15 +02:00
|
|
|
if (size == -1) {
|
|
|
|
s = r_print_format_struct (p, seeki,
|
2014-10-26 02:22:58 +02:00
|
|
|
buf+i, len, structname, slide, json);
|
2014-10-07 17:51:15 +02:00
|
|
|
i+= (isptr) ? 4 : s;
|
|
|
|
} else {
|
|
|
|
p->printf ("[\n");
|
|
|
|
s = r_print_format_struct (p, seeki,
|
2014-10-26 02:22:58 +02:00
|
|
|
buf+i, len, structname, slide, json);
|
2014-10-07 17:51:15 +02:00
|
|
|
i+= (isptr) ? 4 : s;
|
|
|
|
size--;
|
|
|
|
while (size--) {
|
|
|
|
p->printf (",\n");
|
|
|
|
s = r_print_format_struct (p, seeki,
|
2014-10-26 02:22:58 +02:00
|
|
|
buf+i, len, structname, slide, json);
|
2014-10-07 17:51:15 +02:00
|
|
|
i+= (isptr) ? 4 : s;
|
|
|
|
}
|
2014-10-26 02:03:11 +01:00
|
|
|
p->printf ("]");
|
2014-10-07 17:51:15 +02:00
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
oldslide = slide;
|
2014-09-16 00:12:59 +02:00
|
|
|
slide -= (isptr) ? STRUCTPTR : NESTEDSTRUCT;
|
2014-10-26 02:22:58 +02:00
|
|
|
if (flag) {
|
|
|
|
oldslide = slide;
|
|
|
|
slide-=STRUCTFLAG;
|
|
|
|
}
|
2014-09-21 01:39:18 +02:00
|
|
|
free (osn);
|
2014-08-07 23:59:43 +02:00
|
|
|
break;
|
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
default:
|
|
|
|
/* ignore unknown chars */
|
2014-07-14 15:05:45 +02:00
|
|
|
invalid = 1;
|
2009-03-12 00:42:32 +00:00
|
|
|
break;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-10-26 02:22:58 +02:00
|
|
|
if (!json && !flag && (!MUSTSEE || MUSTSET))
|
2014-08-02 02:19:43 +02:00
|
|
|
idx++;
|
2011-05-20 20:47:40 +02:00
|
|
|
if (viewflags && p->offname) {
|
2011-05-20 19:49:50 +02:00
|
|
|
const char *s = p->offname (p->user, seeki);
|
2014-08-02 02:19:43 +02:00
|
|
|
if (s)
|
|
|
|
p->printf ("@(%s)", s);
|
2011-05-20 19:49:50 +02:00
|
|
|
s = p->offname (p->user, addr);
|
2014-08-02 02:19:43 +02:00
|
|
|
if (s)
|
|
|
|
p->printf ("*(%s)", s);
|
2011-05-20 19:49:50 +02:00
|
|
|
}
|
2014-08-07 23:59:43 +02:00
|
|
|
if (tmp != 'D' && !invalid && name==NULL)
|
2013-02-25 09:36:07 +01:00
|
|
|
p->printf ("\n");
|
2011-05-20 19:49:50 +02:00
|
|
|
last = tmp;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-10-26 13:50:58 +01:00
|
|
|
if (otimes>1) {
|
|
|
|
if (json) p->printf ("]");
|
2014-10-26 02:22:58 +02:00
|
|
|
else p->printf ("}\n");
|
2014-10-26 13:50:58 +01:00
|
|
|
}
|
2009-02-05 22:08:46 +01:00
|
|
|
arg = orig;
|
2014-10-26 02:22:58 +02:00
|
|
|
oldslide = 0;
|
|
|
|
// if (json && arg != argend && slide>=oldslide) p->printf (",\n");
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|
2014-10-26 02:03:11 +01:00
|
|
|
if (json && slide==0) p->printf("]");
|
2013-02-01 03:15:48 +01:00
|
|
|
if (oldprintf)
|
|
|
|
p->printf = oldprintf;
|
2013-08-11 14:23:51 +02:00
|
|
|
beach:
|
|
|
|
free (buf);
|
2014-08-02 02:19:43 +02:00
|
|
|
if (args)
|
2014-04-25 04:58:48 +04:00
|
|
|
free (args);
|
2013-01-22 05:06:12 +01:00
|
|
|
return i;
|
2009-02-05 22:08:46 +01:00
|
|
|
}
|