radare2/libr/util/format.c

2474 lines
58 KiB
C
Raw Normal View History

2018-04-08 22:10:46 +02:00
/* radare - LGPL - Copyright 2007-2018 - pancake & Skia */
#include "r_cons.h"
#include "r_util.h"
2018-11-14 12:56:02 +01:00
#include "r_util/r_print.h"
#include "r_reg.h"
2017-05-09 23:53:20 +02:00
#ifdef _MSC_VER
#include <time.h>
#endif
#define NOPTR 0
#define PTRSEEK 1
#define PTRBACK 2
#define NULLPTR 3
#define STRUCTPTR 100
#define NESTEDSTRUCT 1
#define STRUCTFLAG 10000
#define NESTDEPTH 14
#define ARRAYINDEX_COEF 10000
#define MUSTSEE (mode & R_PRINT_MUSTSEE && mode & R_PRINT_ISFIELD && !(mode & R_PRINT_JSON))
#define MUSTSET (mode & R_PRINT_MUSTSET && mode & R_PRINT_ISFIELD && setval)
#define SEEVALUE (mode & R_PRINT_VALUE)
#define MUSTSEEJSON (mode & R_PRINT_JSON && mode & R_PRINT_ISFIELD)
2016-11-03 11:59:50 +01:00
//this define is used as a way to acknowledge when updateAddr should take len
//as real len of the buffer
#define THRESHOLD (-4444)
2016-11-03 11:59:50 +01:00
//TODO REWRITE THIS IS BECOMING A NIGHTMARE
2016-10-30 23:57:25 +01:00
static float updateAddr(const ut8 *buf, int len, int endian, ut64 *addr, ut64 *addr64) {
float f = 0.0;
2016-08-02 00:37:07 +02:00
// assert sizeof (float) == sizeof (ut32))
ut32 tmpaddr;
2016-10-31 00:43:24 +01:00
// XXX 999 is used as an implicit buffer size, we should pass the buffer size to every function too, otherwise this code will give us some problems
2016-11-03 11:59:50 +01:00
if (len >= THRESHOLD - 7 && len < THRESHOLD) {
len = len + THRESHOLD; // get the real len to avoid oob
} else {
len = 999;
}
2016-10-30 23:57:25 +01:00
if (len >= sizeof (float)) {
r_mem_swaporcopy ((ut8*)&f, buf, sizeof (float), endian);
}
if (addr && len > 3) {
tmpaddr = r_read_ble32 (buf, endian);
*addr = (ut64)tmpaddr;
}
2016-10-30 23:57:25 +01:00
if (addr64 && len > 7) {
*addr64 = r_read_ble64 (buf, endian);
}
2016-08-02 00:37:07 +02:00
return f;
2014-08-01 00:15:17 +02:00
}
static int r_get_size(RNum *num, ut8 *buf, int endian, const char *s) {
2016-08-02 00:37:07 +02:00
int size = 0, len = strlen (s);
ut64 addr;
if (s[0] == '*' && len >= 4) { // value pointed by the address
2016-08-02 00:37:07 +02:00
int offset = (int)r_num_math (num, s + 1);
2016-10-31 00:43:24 +01:00
(void)updateAddr (buf + offset, 999, endian, &addr, NULL);
return addr;
} else {
2016-08-02 00:37:07 +02:00
// flag handling doesnt seems to work here
size = r_num_math (num, s);
}
return size;
}
static void r_print_format_quadword(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
ut64 addr64;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
2016-10-31 00:43:24 +01:00
elem = size / ARRAYINDEX_COEF - 1;
size %= ARRAYINDEX_COEF;
}
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, NULL, &addr64);
if (MUSTSET) {
p->cb_printf ("wv8 %s @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem*8:0));
} else if (MUSTSEE) {
if (!SEEVALUE) {
2016-06-01 12:13:09 +02:00
p->cb_printf ("0x%08"PFMT64x" = (qword)",
seeki + ((elem >= 0)? elem * 2: 0));
}
2016-08-02 00:37:07 +02:00
if (size == -1) {
p->cb_printf ("0x%016"PFMT64x, addr64);
2016-06-01 12:13:09 +02:00
} else {
2016-08-02 00:37:07 +02:00
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, NULL, &addr64);
if (elem == -1 || elem == 0) {
p->cb_printf ("0x%016"PFMT64x, addr64);
if (elem == 0) {
elem = -2;
}
}
2016-06-01 12:13:09 +02:00
if (size != 0 && elem == -1) {
p->cb_printf (", ");
2016-06-01 12:13:09 +02:00
}
if (elem > -1) {
elem--;
}
i += 8;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
2014-10-26 02:22:58 +02:00
}
} else if (MUSTSEEJSON) {
2016-05-30 15:30:55 +02:00
if (size==-1) {
2016-03-08 11:44:32 +01:00
p->cb_printf ("%"PFMT64d, addr64);
2016-05-30 15:30:55 +02:00
} else {
p->cb_printf ("[ ");
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, NULL, &addr64);
if (elem == -1 || elem == 0) {
2016-03-08 11:44:32 +01:00
p->cb_printf ("%"PFMT64d, addr64);
if (elem == 0) {
elem = -2;
}
}
2016-05-30 15:30:55 +02:00
if (size != 0 && elem == -1) {
p->cb_printf (", ");
2016-05-30 15:30:55 +02:00
}
if (elem > -1) {
elem--;
}
2016-05-30 15:30:55 +02:00
i += 8;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
2014-08-01 19:33:52 +02:00
}
}
static void r_print_format_byte(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
2016-08-02 00:37:07 +02:00
elem = size / ARRAYINDEX_COEF - 1;
size %= ARRAYINDEX_COEF;
}
if (MUSTSET) {
2016-08-02 00:37:07 +02:00
p->cb_printf ("\"w %s\" @ 0x%08"PFMT64x"\n", setval, seeki + ((elem >= 0) ? elem : 0));
} else if (MUSTSEE) {
2016-08-02 00:37:07 +02:00
if (!SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ", seeki + ((elem >= 0) ? elem : 0));
}
if (size == -1) {
p->cb_printf ("0x%02x", buf[i]);
2016-06-01 12:13:09 +02:00
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
if (elem == -1 || elem == 0) {
p->cb_printf ("0x%02x", buf[i]);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i++;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
} else if (MUSTSEEJSON) {
2016-08-02 00:37:07 +02:00
if (size==-1) {
p->cb_printf ("%d", buf[i]);
2016-08-02 00:37:07 +02:00
} else {
p->cb_printf ("[ ");
while (size--) {
if (elem == -1 || elem == 0) {
p->cb_printf (", %d", buf[i]);
if (elem == 0) {
elem = -2;
}
}
2016-08-02 00:37:07 +02:00
if (size != 0 && elem == -1) {
p->cb_printf (", ");
2016-08-02 00:37:07 +02:00
}
if (elem > -1) {
elem--;
}
i++;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
2014-08-01 19:33:52 +02:00
}
}
2018-08-07 00:46:26 -07:00
// Return number of consumed bytes
2015-05-02 15:07:08 +02:00
static int r_print_format_uleb(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
int elem = -1;
2018-08-07 00:46:26 -07:00
int s = 0, offset = 0;
ut64 value = 0;
2015-05-02 15:07:08 +02:00
if (size >= ARRAYINDEX_COEF) {
2016-06-01 12:13:09 +02:00
elem = size / ARRAYINDEX_COEF - 1;
2015-05-02 15:07:08 +02:00
size %= ARRAYINDEX_COEF;
}
if (MUSTSET) {
ut8 *tmp;
char *nbr;
do {
2018-08-07 00:46:26 -07:00
r_uleb128_decode (buf+i, &s, &value);
i += s;
2015-05-02 15:07:08 +02:00
offset += s;
} while (elem--);
tmp = (ut8*) r_uleb128_encode (r_num_math (NULL, setval), &s);
nbr = r_hex_bin2strdup (tmp, s);
2018-08-07 00:46:26 -07:00
p->cb_printf ("\"wx %s\" @ 0x%08"PFMT64x"\n", nbr, seeki+offset-s);
2015-05-02 15:07:08 +02:00
free (tmp);
free (nbr);
} else if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki);
}
2015-05-02 15:07:08 +02:00
if (size==-1) {
2018-08-07 00:46:26 -07:00
r_uleb128_decode (buf+i, &offset, &value);
p->cb_printf ("%"PFMT64d, value);
2015-05-02 15:07:08 +02:00
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
2015-05-02 15:07:08 +02:00
while (size--) {
if (elem == -1 || elem == 0) {
r_uleb128_decode (buf+i, &s, &value);
2018-08-07 00:46:26 -07:00
i += s;
offset += s;
p->cb_printf ("%"PFMT64d, value);
if (elem == 0) {
elem = -2;
}
2015-05-02 15:07:08 +02:00
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
}
if (!SEEVALUE) {
p->cb_printf (" ]");
2015-05-02 15:07:08 +02:00
}
}
} else if (MUSTSEEJSON) {
if (size==-1) {
2018-08-07 00:46:26 -07:00
r_uleb128_decode (buf+i, &offset, &value);
p->cb_printf ("\"%"PFMT64d"\"", value);
2015-05-02 15:07:08 +02:00
} else {
p->cb_printf ("[ ");
2015-05-02 15:07:08 +02:00
while (size--) {
if (elem == -1 || elem == 0) {
r_uleb128_decode (buf+i, &s, &value);
2018-08-07 00:46:26 -07:00
i += s;
offset += s;
p->cb_printf ("\"%"PFMT64d"\"", value);
if (elem == 0) {
elem = -2;
}
2015-05-02 15:07:08 +02:00
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
2015-05-02 15:07:08 +02:00
}
p->cb_printf (" ]");
2015-05-02 15:07:08 +02:00
}
p->cb_printf ("}");
2015-05-02 15:07:08 +02:00
}
2018-08-07 00:46:26 -07:00
return offset;
2015-05-02 15:07:08 +02:00
}
static void r_print_format_char(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
if (MUSTSET) {
p->cb_printf ("\"w %s\" @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem:0));
} else if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki + ((elem >= 0) ? elem * 2 : 0)); //XXX:: shouldn't it be elem*1??
}
if (size == -1) {
p->cb_printf ("'%c'", IS_PRINTABLE (buf[i])?buf[i]:'.');
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
if (elem == -1 || elem == 0) {
p->cb_printf ("'%c'", IS_PRINTABLE (buf[i])?buf[i]:'.');
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i++;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
} else if (MUSTSEEJSON) {
if (size == -1) {
p->cb_printf ("\"%c\"", IS_PRINTABLE (buf[i])?buf[i]:'.');
} else {
p->cb_printf ("[ ");
while (size--) {
if (elem == -1 || elem == 0) {
p->cb_printf ("\"%c\"", IS_PRINTABLE (buf[i])?buf[i]:'.');
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i++;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
2014-08-01 19:33:52 +02:00
}
}
2015-03-03 01:03:49 +01:00
static void r_print_format_decchar(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
if (MUSTSET) {
p->cb_printf ("\"w %s\" @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem:0));
2015-03-03 01:03:49 +01:00
} else if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki + ((elem >= 0) ? elem : 0));
}
if (size == -1) {
p->cb_printf ("%d", buf[i]);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
2015-03-03 01:03:49 +01:00
while (size--) {
if (elem == -1 || elem == 0) {
p->cb_printf ("%d", buf[i]);
if (elem == 0) {
elem = -2;
}
2015-03-03 01:03:49 +01:00
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
2015-03-03 01:03:49 +01:00
i++;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
2015-03-03 01:03:49 +01:00
}
} else if (MUSTSEEJSON) {
if (size == -1) {
p->cb_printf ("\"%d\"", buf[i]);
} else {
p->cb_printf ("[ ");
2015-03-03 01:03:49 +01:00
while (size--) {
if (elem == -1 || elem == 0) {
p->cb_printf ("\"%d\"", buf[i]);
if (elem == 0) {
elem = -2;
}
2015-03-03 01:03:49 +01:00
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
2015-03-03 01:03:49 +01:00
i++;
}
p->cb_printf (" ]");
2015-03-03 01:03:49 +01:00
}
p->cb_printf ("}");
2015-03-03 01:03:49 +01:00
}
}
static int r_print_format_string(const RPrint* p, ut64 seeki, ut64 addr64, ut64 addr, int is64, int mode) {
2014-08-01 19:33:52 +02:00
ut8 buffer[255];
2014-12-01 23:30:32 +01:00
buffer[0] = 0;
if (!p->iob.read_at) {
eprintf ("(cannot read memory)\n");
2014-08-01 19:33:52 +02:00
return -1;
}
int res = (is64 == 1)
? p->iob.read_at (p->iob.io, addr64, buffer, sizeof (buffer) - 8)
: p->iob.read_at (p->iob.io, (ut64)addr, buffer, sizeof (buffer) - 8);
if (MUSTSEEJSON) {
char *encstr = r_str_utf16_encode ((const char *)buffer, -1);
if (encstr) {
p->cb_printf ("%d,\"string\":\"%s\"}", seeki, encstr);
free (encstr);
}
} else if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki);
}
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " -> 0x%08" PFMT64x " ", seeki, addr);
}
if (res && buffer[0] != 0xff && buffer[1] != 0xff) {
p->cb_printf ("%s", buffer);
}
2014-10-26 02:22:58 +02:00
}
2014-08-01 19:33:52 +02:00
return 0;
}
static void r_print_format_time(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
2014-10-31 02:09:08 +01:00
ut64 addr;
struct tm timestruct;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (MUSTSET) {
p->cb_printf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem*4:0));
} else if (MUSTSEE) {
char *timestr = strdup(asctime (gmtime_r ((time_t*)&addr, &timestruct)));
*(timestr+24) = '\0';
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki + ((elem >= 0) ? elem * 4 : 0));
}
if (size==-1) {
p->cb_printf ("%s", timestr);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
2014-10-31 02:09:08 +01:00
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
free (timestr);
timestr = strdup (asctime (gmtime_r ((time_t*)&addr, &timestruct)));
*(timestr+24) = '\0';
if (elem == -1 || elem == 0) {
p->cb_printf ("%s", timestr);
2016-10-31 00:43:24 +01:00
if (elem == 0) {
elem = -2;
}
}
2016-10-31 00:43:24 +01:00
if (size != 0 && elem == -1) {
p->cb_printf (", ");
2016-10-31 00:43:24 +01:00
}
if (elem > -1) {
elem--;
}
i += 4;
2014-10-31 02:09:08 +01:00
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
2014-10-31 02:09:08 +01:00
}
free (timestr);
} else if (MUSTSEEJSON) {
char *timestr = strdup (asctime (gmtime_r ((time_t*)&addr, &timestruct)));
*(timestr+24) = '\0';
if (size==-1) {
p->cb_printf ("\"%s\"", timestr);
} else {
p->cb_printf ("[ ");
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
free (timestr);
timestr = strdup (asctime (gmtime_r ((time_t*)&addr, &timestruct)));
*(timestr+24) = '\0';
if (elem == -1 || elem == 0) {
p->cb_printf ("\"%s\"", timestr);
if (elem == 0) {
elem = -2;
}
}
2016-10-31 00:43:24 +01:00
if (size != 0 && elem == -1) {
p->cb_printf (", ");
2016-10-31 00:43:24 +01:00
}
if (elem > -1) {
elem--;
}
i += 4;
}
p->cb_printf (" ]");
}
free (timestr);
p->cb_printf ("}");
2014-10-31 02:09:08 +01:00
}
}
2014-08-01 19:33:52 +02:00
// TODO: support unsigned int?
static void r_print_format_hex(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
ut64 addr;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (MUSTSET) {
p->cb_printf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem*4:0));
} else if (mode & R_PRINT_DOT) {
p->cb_printf ("%"PFMT64d, addr);
} else if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki + ((elem >= 0) ? elem * 4 : 0));
}
if (size==-1) {
p->cb_printf ("%"PFMT64d, addr);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (elem == -1 || elem == 0) {
p->cb_printf ("%"PFMT64d, addr);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i += 4;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
} else if (MUSTSEEJSON) {
if (size == -1) {
p->cb_printf ("%d", addr);
} else {
p->cb_printf ("[ ");
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (elem == -1 || elem == 0) {
p->cb_printf ("%d", addr);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i+=4;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
2014-08-01 19:33:52 +02:00
}
}
2017-01-12 00:31:53 +01:00
static void r_print_format_int(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
ut64 addr;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (MUSTSET) {
p->cb_printf ("wv4 %s @ %"PFMT64d"\n", setval, seeki+((elem>=0)?elem*4:0));
} else if (mode & R_PRINT_DOT) {
p->cb_printf ("0x%08"PFMT64x, addr);
} else if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ", seeki+((elem>=0)?elem*4:0));
}
if (size == -1) {
p->cb_printf ("%"PFMT64d, (st64)(st32)addr);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (elem == -1 || elem == 0) {
p->cb_printf ("%"PFMT64d, (st64)(st32)addr);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i += 4;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
} else if (MUSTSEEJSON) {
if (size == -1) {
2017-01-12 00:31:53 +01:00
p->cb_printf ("%d", addr);
} else {
2017-01-12 00:31:53 +01:00
p->cb_printf ("[ ");
while (size--) {
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (elem == -1 || elem == 0) {
p->cb_printf ("%d", addr);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i+=4;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
}
}
2018-04-08 22:10:46 +02:00
2016-06-06 10:36:09 +02:00
static int r_print_format_disasm(const RPrint* p, ut64 seeki, int size) {
ut64 prevseeki = seeki;
2018-04-08 22:10:46 +02:00
if (!p->disasm || !p->user) {
return 0;
}
2016-06-06 10:36:09 +02:00
size = R_MAX (1, size);
while (size-- > 0) {
seeki += p->disasm (p->user, seeki);
}
return seeki - prevseeki;
}
static void r_print_format_octal (const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
ut64 addr;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (MUSTSET) {
p->cb_printf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem*4:0));
2015-10-25 03:28:29 +01:00
} else if (mode & R_PRINT_DOT) {
p->cb_printf ("0%"PFMT64o, addr);
} else if (MUSTSEE) {
ut32 addr32 = (ut32)addr;
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki + ((elem >= 0) ? elem * 4 : 0));
}
if (!SEEVALUE) {
p->cb_printf ("(octal) ");
}
if (size == -1) {
2016-06-01 12:13:09 +02:00
p->cb_printf (" 0%08"PFMT64o, addr32);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
addr32 = (ut32)addr;
if (elem == -1 || elem == 0) {
p->cb_printf ("0%08"PFMT64o, addr32);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i+=4;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
2014-10-26 02:22:58 +02:00
}
} else if (MUSTSEEJSON) {
ut32 addr32 = (ut32)addr;
if (size == -1) {
p->cb_printf ("%d", addr32);
} else {
p->cb_printf ("[ ");
while (size--) {
updateAddr (buf, i, endian, &addr, NULL);
addr32 = (ut32)addr;
if (elem == -1 || elem == 0) {
p->cb_printf ("%d", addr32);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i+=4;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
}
}
static void r_print_format_hexflag(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
2016-10-30 23:57:25 +01:00
ut64 addr = 0;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
if (MUSTSET) {
p->cb_printf ("wv4 %s @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem*4:0));
} else if (mode & R_PRINT_DOT) {
p->cb_printf ("0x%08"PFMT64x, addr & UT32_MAX);
} else if (MUSTSEE) {
2014-08-01 19:33:52 +02:00
ut32 addr32 = (ut32)addr;
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki + ((elem >= 0) ? elem * 4 : 0));
}
if (size==-1) {
2016-10-30 23:57:25 +01:00
p->cb_printf ("0x%08"PFMT64x, (ut64)addr32);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
addr32 = (ut32)addr;
if (elem == -1 || elem == 0) {
p->cb_printf ("0x%08"PFMT64x, addr32);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i+=4;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
} else if (MUSTSEEJSON) {
ut32 addr32 = (ut32)addr;
if (size == -1) {
p->cb_printf ("%d", addr32);
} else {
p->cb_printf ("[ ");
while (size--) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
addr32 = (ut32)addr;
if (elem == -1 || elem == 0) {
p->cb_printf ("%d", addr32);
if (elem == 0) {
elem = -2;
}
}
2016-10-31 00:43:24 +01:00
if (size != 0 && elem == -1) {
p->cb_printf (",");
2016-10-31 00:43:24 +01:00
}
if (elem > -1) {
elem--;
}
i += 4;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
2014-08-01 19:33:52 +02:00
}
}
static int r_print_format_10bytes(const RPrint* p, int mode, const char* setval,
ut64 seeki, ut64 addr, ut8* buf) {
2014-08-01 19:33:52 +02:00
ut8 buffer[255];
int j;
if (MUSTSET) {
p->cb_printf ("?e pf B not yet implemented\n");
2015-10-25 03:28:29 +01:00
} else if (mode & R_PRINT_DOT) {
2016-09-19 02:09:42 +02:00
for (j = 0; j<10; j++) {
2015-10-25 03:28:29 +01:00
p->cb_printf ("%02x ", buf[j]);
2016-09-19 02:09:42 +02:00
}
} else if (MUSTSEE) {
2014-08-01 19:33:52 +02:00
if (!p->iob.read_at) {
printf ("(cannot read memory)\n");
return -1;
2016-06-01 12:13:09 +02:00
}
p->iob.read_at (p->iob.io, (ut64)addr, buffer, 248);
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki);
}
2016-09-19 02:09:42 +02:00
for (j=0; j<10; j++) {
p->cb_printf ("%02x ", buf[j]);
2016-09-19 02:09:42 +02:00
}
if (!SEEVALUE) {
p->cb_printf (" ... (");
}
2016-09-19 02:09:42 +02:00
for (j = 0; j < 10; j++) {
if (!SEEVALUE) {
2016-09-19 02:09:42 +02:00
if (IS_PRINTABLE (buf[j])) {
p->cb_printf ("%c", buf[j]);
2016-09-19 02:09:42 +02:00
} else {
p->cb_printf (".");
2016-09-19 02:09:42 +02:00
}
}
}
if (!SEEVALUE) {
p->cb_printf (")");
}
} else if (MUSTSEEJSON) {
if (!p->iob.read_at) {
printf ("(cannot read memory)\n");
return -1;
2016-09-19 02:09:42 +02:00
} else {
p->iob.read_at (p->iob.io, (ut64)addr, buffer, 248);
2016-09-19 02:09:42 +02:00
}
p->cb_printf ("[ %d", buf[0]);
2016-09-19 02:09:42 +02:00
j = 1;
for (; j < 10; j++) {
p->cb_printf (", %d", buf[j]);
2016-09-19 02:09:42 +02:00
}
p->cb_printf ("]");
return 0;
2014-08-01 19:33:52 +02:00
}
return 0;
}
static int r_print_format_hexpairs(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
2014-08-02 02:19:43 +02:00
int j;
2016-09-19 02:09:42 +02:00
size = (size == -1) ? 1 : size;
if (MUSTSET) {
p->cb_printf ("?e pf X not yet implemented\n");
2015-10-25 03:28:29 +01:00
} else if (mode & R_PRINT_DOT) {
2016-09-19 02:09:42 +02:00
for (j = 0; j < size; j++) {
2016-06-01 12:13:09 +02:00
p->cb_printf ("%02x", buf[i + j]);
2016-09-19 02:09:42 +02:00
}
} else if (MUSTSEE) {
2014-08-02 02:19:43 +02:00
size = (size < 1) ? 1 : size;
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki);
}
2016-06-01 12:13:09 +02:00
for (j = 0; j < size; j++) {
p->cb_printf ("%02x ", buf[i + j]);
}
if (!SEEVALUE) {
p->cb_printf (" ... (");
}
for (j=0; j<size; j++) {
if (!SEEVALUE) {
if (IS_PRINTABLE (buf[j])) {
p->cb_printf ("%c", buf[i+j]);
} else {
p->cb_printf (".");
}
}
}
p->cb_printf (")");
} else if (MUSTSEEJSON) {
size = (size < 1) ? 1 : size;
p->cb_printf ("[ %d", buf[0]);
2016-06-01 12:13:09 +02:00
j = 1;
for (; j < 10; j++) {
p->cb_printf (", %d", buf[j]);
}
p->cb_printf ("]}");
return size;
2014-08-02 02:19:43 +02:00
}
return size;
}
static void r_print_format_float(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
2016-08-02 00:37:07 +02:00
float val_f = 0.0f;
ut64 addr = 0;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
2016-08-02 00:37:07 +02:00
elem = size/ARRAYINDEX_COEF - 1;
size %= ARRAYINDEX_COEF;
}
2016-10-31 00:43:24 +01:00
val_f = updateAddr (buf + i, 999, endian, &addr, NULL);
if (MUSTSET) {
2016-08-02 00:37:07 +02:00
p->cb_printf ("wv4 %s @ 0x%08"PFMT64x"\n", setval,
seeki + ((elem >= 0) ? elem * 4 : 0));
2015-10-25 03:28:29 +01:00
} else if (mode & R_PRINT_DOT) {
2016-08-02 00:37:07 +02:00
p->cb_printf ("%f", val_f);
} else {
2016-08-02 00:37:07 +02:00
if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ",
seeki + ((elem >= 0) ? elem * 4 : 0));
}
}
if (size == -1) {
p->cb_printf ("%f", val_f);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
2016-10-31 00:43:24 +01:00
val_f = updateAddr (buf + i, 9999, endian, &addr, NULL);
if (elem == -1 || elem == 0) {
2016-08-02 00:37:07 +02:00
p->cb_printf ("%f", val_f);
if (elem == 0) {
elem = -2;
}
}
2016-08-02 00:37:07 +02:00
if (size != 0 && elem == -1) {
p->cb_printf (", ");
2016-08-02 00:37:07 +02:00
}
if (elem > -1) {
elem--;
}
i += 4;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
2016-08-02 00:37:07 +02:00
if (MUSTSEEJSON) {
p->cb_printf ("}");
}
}
}
static void r_print_format_double(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
double val_f = 0.0;
ut64 addr = 0;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF - 1;
size %= ARRAYINDEX_COEF;
}
updateAddr (buf + i, 999, endian, &addr, NULL);
r_mem_swaporcopy ((ut8*)&val_f, buf + i, sizeof (double), endian);
if (MUSTSET) {
p->cb_printf ("wv8 %s @ 0x%08"PFMT64x"\n", setval,
seeki + ((elem >= 0) ? elem * 8 : 0));
} else if (mode & R_PRINT_DOT) {
p->cb_printf ("%f", val_f);
} else {
if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ",
seeki + ((elem >= 0) ? elem * 8 : 0));
}
}
if (size == -1) {
p->cb_printf ("%f", val_f);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
2018-04-08 22:10:46 +02:00
// XXX this 999 is scary
updateAddr (buf + i, 9999, endian, &addr, NULL);
r_mem_swaporcopy ((ut8*)&val_f, buf + i, sizeof (double), endian);
if (elem == -1 || elem == 0) {
p->cb_printf ("%f", val_f);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i += 8;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
if (MUSTSEEJSON) {
p->cb_printf ("}");
}
}
}
static void r_print_format_word(const RPrint* p, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
ut64 addr;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
elem = size/ARRAYINDEX_COEF-1;
size %= ARRAYINDEX_COEF;
}
2016-08-02 00:37:07 +02:00
addr = endian
? (*(buf + i)) << 8 | (*(buf + i + 1))
: (*(buf + i + 1)) << 8 | (*(buf + i));
if (MUSTSET) {
p->cb_printf ("wv2 %s @ 0x%08"PFMT64x"\n", setval, seeki+((elem>=0)?elem*2:0));
2015-10-25 03:28:29 +01:00
} else if (mode & R_PRINT_DOT) {
2016-08-02 00:37:07 +02:00
if (size == -1) {
2015-10-25 03:28:29 +01:00
p->cb_printf ("0x%04x", addr);
2016-08-02 00:37:07 +02:00
}
while ((size -= 2) > 0) {
addr = endian
? (*(buf+i))<<8 | (*(buf+i+1))
: (*(buf+i+1))<<8 | (*(buf+i));
2015-10-25 03:28:29 +01:00
if (elem == -1 || elem == 0) {
p->cb_printf ("%d", addr);
if (elem == 0) {
elem = -2;
}
2015-10-25 03:28:29 +01:00
}
2016-08-02 00:37:07 +02:00
if (size != 0 && elem == -1) {
2015-10-25 03:28:29 +01:00
p->cb_printf (",");
2016-08-02 00:37:07 +02:00
}
if (elem > -1) {
elem--;
}
i += 2;
2015-10-25 03:28:29 +01:00
}
} else if (MUSTSEE) {
2016-08-02 00:37:07 +02:00
if (!SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ", seeki+((elem>=0)?elem*2:0));
}
2015-10-25 03:28:29 +01:00
if (size==-1) {
p->cb_printf ("0x%04x", addr);
2015-10-25 03:28:29 +01:00
} else {
2016-08-02 00:37:07 +02:00
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
2016-08-02 00:37:07 +02:00
addr = endian
? (*(buf+i))<<8 | (*(buf+i+1))
: (*(buf+i+1))<<8 | (*(buf+i));
if (elem == -1 || elem == 0) {
p->cb_printf ("0x%04x", addr);
if (elem == 0) {
elem = -2;
}
}
2016-08-02 00:37:07 +02:00
if (size != 0 && elem == -1) {
p->cb_printf (", ");
2016-08-02 00:37:07 +02:00
}
if (elem > -1) {
elem--;
}
2015-10-25 03:28:29 +01:00
i += 2;
}
2016-08-02 00:37:07 +02:00
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
} else if (MUSTSEEJSON) {
2016-08-02 00:37:07 +02:00
if (size==-1) {
p->cb_printf ("%d", addr);
2016-08-02 00:37:07 +02:00
} else {
p->cb_printf ("[ ");
2016-08-02 00:37:07 +02:00
while ( (size -= 2) >0) {
addr = endian
? (*(buf+i))<<8 | (*(buf+i+1))
: (*(buf+i+1))<<8 | (*(buf+i));
if (elem == -1 || elem == 0) {
p->cb_printf ("%d", addr);
if (elem == 0) {
elem = -2;
}
}
2016-08-02 00:37:07 +02:00
if (size != 0 && elem == -1) {
p->cb_printf (",");
2016-08-02 00:37:07 +02:00
}
if (elem > -1) {
elem--;
}
2016-08-02 00:37:07 +02:00
i += 2;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
}
2014-08-02 02:19:43 +02:00
}
static void r_print_format_nulltermstring(const RPrint* p, const int len, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
if (MUSTSET) {
int buflen = strlen ((const char *)buf+seeki), vallen = strlen(setval);
char *newstring, *ons;
newstring = ons = strdup(setval);
if ((newstring[0] == '\"' && newstring[vallen-1] == '\"')
|| (newstring[0] == '\'' && newstring[vallen-1] == '\'')) {
newstring[vallen-1] = '\0';
newstring++;
2016-08-02 00:37:07 +02:00
vallen -= 2;
}
if (vallen > buflen) {
eprintf ("Warning: new string is longer than previous one\n");
}
p->cb_printf ("wx ");
2016-08-02 00:37:07 +02:00
for (i = 0; i < vallen; i++) {
if (i < vallen - 3 && newstring[i] == '\\' && newstring[i + 1] == 'x') {
p->cb_printf ("%c%c", newstring[i + 2], newstring[i + 3]);
i += 3;
} else {
p->cb_printf ("%2x", newstring[i]);
}
}
p->cb_printf (" @ 0x%08"PFMT64x"\n", seeki);
free(ons);
} else if (mode & R_PRINT_DOT) {
int j = i;
p->cb_printf ("\\\"", seeki);
for (; j<len && ((size==-1 || size-- >0) && buf[j]) ; j++) {
char ch = buf[j];
if (ch == '"') {
p->cb_printf ("\\\"");
} else if (IS_PRINTABLE (ch)) {
p->cb_printf ("%c", ch);
} else {
p->cb_printf (".");
}
}
p->cb_printf ("\\\"");
} else if (MUSTSEE) {
int j = i;
if (!SEEVALUE) {
p->cb_printf ("0x%08" PFMT64x " = ", seeki);
}
2016-08-02 00:37:07 +02:00
for (; j < len && ((size == -1 || size-- > 0) && buf[j]) ; j++) {
if (IS_PRINTABLE (buf[j])) {
p->cb_printf ("%c", buf[j]);
2016-08-02 00:37:07 +02:00
} else {
p->cb_printf (".");
}
}
} else if (MUSTSEEJSON) {
char * utf_encoded_buf = NULL;
p->cb_printf ("%d,\"string\":\"", seeki);
utf_encoded_buf = r_str_escape_utf8 ((const char *) buf, true, true);
if (utf_encoded_buf){
p->cb_printf ("%s", utf_encoded_buf);
free (utf_encoded_buf);
}
p->cb_printf ("\"}");
}
}
static void r_print_format_nulltermwidestring(const RPrint* p, const int len, int endian, int mode,
const char* setval, ut64 seeki, ut8* buf, int i, int size) {
if (MUSTSET) {
int vallen = strlen(setval);
char *newstring, *ons;
newstring = ons = strdup(setval);
if ((newstring[0] == '\"' && newstring[vallen-1] == '\"')
|| (newstring[0] == '\'' && newstring[vallen-1] == '\'')) {
2016-08-02 00:37:07 +02:00
newstring[vallen - 1] = '\0';
newstring++;
2016-08-02 00:37:07 +02:00
vallen -= 2;
}
2016-08-02 00:37:07 +02:00
if ((size = strlen (setval)) > r_wstr_clen((char*)(buf+seeki))) {
eprintf ("Warning: new string is longer than previous one\n");
2016-08-02 00:37:07 +02:00
}
p->cb_printf ("ww %s @ 0x%08"PFMT64x"\n", newstring, seeki);
free(ons);
} else if (MUSTSEE) {
int j = i;
2016-08-02 00:37:07 +02:00
if (!SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ", seeki);
}
2015-04-23 02:12:45 +02:00
for (; j<len && ((size==-1 || size-->0) && buf[j]) ; j+=2) {
2016-08-02 00:37:07 +02:00
if (IS_PRINTABLE (buf[j])) {
p->cb_printf ("%c", buf[j]);
2016-08-02 00:37:07 +02:00
} else {
p->cb_printf (".");
}
}
2018-11-07 11:41:27 +00:00
} else if (MUSTSEEJSON) {
int j = i;
p->cb_printf ("%d,\"string\":\"", seeki);
for (; j < len && ((size == -1 || size-- > 0) && buf[j]); j += 2) {
if (IS_PRINTABLE (buf[j])) {
p->cb_printf ("%c", buf[j]);
} else {
p->cb_printf (".");
}
}
p->cb_printf ("\"}");
}
}
static void r_print_format_bitfield(const RPrint* p, ut64 seeki, char* fmtname,
char* fieldname, ut64 addr, int mode, int size) {
char *bitfield = NULL;
switch (size) {
2016-08-02 00:37:07 +02:00
case 1: addr &= UT8_MAX; break;
case 2: addr &= UT16_MAX; break;
case 4: addr &= UT32_MAX; break;
}
2016-08-02 00:37:07 +02:00
if (MUSTSEE && !SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ", seeki);
}
bitfield = r_type_enum_getbitfield (p->sdb_types, fmtname, addr);
if (bitfield && *bitfield) {
if (MUSTSEEJSON) {
p->cb_printf ("\"%s\"}", bitfield);
} else if (MUSTSEE) {
p->cb_printf (" %s (bitfield) = %s\n", fieldname, bitfield);
}
} else {
if (MUSTSEEJSON) {
p->cb_printf ("\"`tb %s 0x%x`\"}", fmtname, addr);
} else if (MUSTSEE) {
p->cb_printf (" %s (bitfield) = `tb %s 0x%x`\n",
fieldname, fmtname, addr);
}
}
free (bitfield);
}
static void r_print_format_enum (const RPrint* p, ut64 seeki, char* fmtname,
char* fieldname, ut64 addr, int mode, int size) {
char *enumvalue = NULL;
switch (size) {
2016-08-02 00:37:07 +02:00
case 1: addr &= UT8_MAX; break;
case 2: addr &= UT16_MAX; break;
case 4: addr &= UT32_MAX; break;
}
2016-09-06 04:29:42 +02:00
if (MUSTSEE && !SEEVALUE) {
p->cb_printf ("0x%08"PFMT64x" = ", seeki);
}
enumvalue = r_type_enum_member (p->sdb_types, fmtname, NULL, addr);
if (enumvalue && *enumvalue) {
if (mode & R_PRINT_DOT) {
p->cb_printf ("%s.%s", fmtname, enumvalue);
2016-09-06 04:29:42 +02:00
} else if (MUSTSEEJSON) {
p->cb_printf ("\"%s\"}", fmtname);
} else if (MUSTSEE) {
p->cb_printf ("%s (enum %s) = 0x%"PFMT64x" ; %s\n",
fieldname, fmtname, addr, enumvalue);
}
} else {
2016-09-06 04:29:42 +02:00
if (MUSTSEEJSON) {
p->cb_printf ("\"`te %s 0x%x`\"}", fmtname, addr);
} else if (MUSTSEE) {
p->cb_printf ("%s (enum %s) = 0x%x\n",//`te %s 0x%x`\n",
fieldname, fmtname, addr); //enumvalue); //fmtname, addr);
}
}
free (enumvalue);
}
static void r_print_format_register (const RPrint* p, int mode,
const char *name, const char* setval) {
2017-05-28 01:47:37 +02:00
if (!p || !p->get_register || !p->reg) {
return;
}
2015-07-24 17:04:33 +02:00
RRegItem *ri = p->get_register (p->reg, name, R_REG_TYPE_ALL);
2016-05-30 15:30:55 +02:00
if (ri) {
if (MUSTSET) {
p->cb_printf ("dr %s=%s\n", name, setval);
} else if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("%s : 0x%08" PFMT64x "\n", ri->name, p->get_register_value (p->reg, ri));
} else {
p->cb_printf ("0x%08" PFMT64x "\n", p->get_register_value (p->reg, ri));
}
2016-05-30 15:30:55 +02:00
} else if (MUSTSEEJSON) {
p->cb_printf ("%d}", p->get_register_value (p->reg, ri));
}
} else {
p->cb_printf ("Register %s does not exists\n", name);
}
}
static void r_print_format_num_specifier (const RPrint *p, ut64 addr, int bytes, int sign) {
#define EXT(T) (sign ? (signed T)(addr) : (unsigned T)(addr) )
const char *fs64 = sign ? "%"PFMT64d : "%"PFMT64u;
const char *fs = sign ? "%d" : "%u";
if (bytes == 1) {
p->cb_printf (fs, EXT(char));
} else if (bytes == 2) {
p->cb_printf (fs, EXT(short));
} else if (bytes == 4) {
p->cb_printf (fs, EXT(int)); //XXX: int is not necessarily 4 bytes I guess.
} else if (bytes == 8) {
p->cb_printf (fs64, addr);
}
#undef EXT
}
static void r_print_format_num (const RPrint *p, int endian, int mode, const char *setval, ut64 seeki, ut8 *buf, int i, int bytes, int sign, int size) {
2016-11-08 15:37:44 +01:00
ut64 addr = 0LL;
int elem = -1;
if (size >= ARRAYINDEX_COEF) {
2016-10-30 23:57:25 +01:00
elem = size / ARRAYINDEX_COEF - 1;
size %= ARRAYINDEX_COEF;
}
if (bytes == 8) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, NULL, &addr);
} else {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
}
if (MUSTSET) {
p->cb_printf ("wv%d %s @ 0x%08"PFMT64x"\n", bytes, setval, seeki+((elem>=0)?elem*(bytes):0));
} else if (mode & R_PRINT_DOT) {
p->cb_printf ("%"PFMT64u, addr);
} else if (MUSTSEE) {
2016-10-30 23:57:25 +01:00
if (!SEEVALUE) {
2016-11-08 15:37:44 +01:00
p->cb_printf ("0x%08"PFMT64x" = ", seeki + ((elem >= 0)? elem * bytes: 0));
2016-10-30 23:57:25 +01:00
}
if (size == -1) {
r_print_format_num_specifier (p, addr, bytes, sign);
} else {
if (!SEEVALUE) {
p->cb_printf ("[ ");
}
while (size--) {
if (bytes == 8) {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, NULL, &addr);
} else {
2016-10-31 00:43:24 +01:00
updateAddr (buf + i, size - i, endian, &addr, NULL);
}
if (elem == -1 || elem == 0) {
r_print_format_num_specifier (p, addr, bytes, sign);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i += bytes;
}
if (!SEEVALUE) {
p->cb_printf (" ]");
}
}
} else if (MUSTSEEJSON) {
if (size == -1) {
r_print_format_num_specifier (p, addr, bytes, sign);
} else {
p->cb_printf ("[ ");
while (size--) {
if (bytes == 8) {
2016-10-30 23:57:25 +01:00
updateAddr (buf + i, size, endian, NULL, &addr);
} else {
2016-10-30 23:57:25 +01:00
updateAddr (buf + i, size, endian, &addr, NULL);
}
if (elem == -1 || elem == 0) {
r_print_format_num_specifier (p, addr, bytes, sign);
if (elem == 0) {
elem = -2;
}
}
if (size != 0 && elem == -1) {
p->cb_printf (", ");
}
if (elem > -1) {
elem--;
}
i += bytes;
}
p->cb_printf (" ]");
}
p->cb_printf ("}");
}
}
// XXX: this is somewhat incomplete. must be updated to handle all format chars
2017-09-26 23:45:42 +02:00
int r_print_format_struct_size(const char *f, RPrint *p, int mode, int n) {
char *end, *args, *fmt;
int size = 0, tabsize = 0, i, idx = 0, biggest = 0, fmt_len = 0;
bool tabsize_set = false;
2016-08-24 22:02:38 +02:00
if (!f) {
return -1;
}
if (n >= 5) { // This is the nesting level, is this not a bit arbitrary?!
2017-09-26 23:45:42 +02:00
return 0;
}
char *o = strdup (f);
2016-08-24 22:02:38 +02:00
if (!o) {
return -1;
}
2016-02-26 15:35:02 -05:00
end = strchr (o, ' ');
fmt = o;
if (!end && !(end = strchr (o, '\0'))) {
free (o);
2014-08-31 02:09:35 +02:00
return -1;
}
2014-11-26 01:03:48 +01:00
if (*end) {
*end = 0;
2016-05-30 15:30:55 +02:00
args = strdup (end + 1);
2014-11-26 01:03:48 +01:00
} else {
args = strdup ("");
}
2015-02-05 01:59:32 +01:00
if (fmt[0] == '0') {
mode |= R_PRINT_UNIONMODE;
fmt++;
} else {
mode &= ~R_PRINT_UNIONMODE;
}
i = 0;
2017-02-06 00:50:54 +03:00
if (IS_DIGIT(fmt[i])) {
while (IS_DIGIT(fmt[i])) {
i++;
}
}
int words = r_str_word_set0_stack (args);
fmt_len = strlen (fmt);
for (; i < fmt_len; i++) {
2014-10-07 17:51:15 +02:00
if (fmt[i] == '[') {
2016-08-24 22:02:38 +02:00
char *end = strchr (fmt + i,']');
if (!end) {
2014-10-07 17:51:15 +02:00
eprintf ("No end bracket.\n");
continue;
}
*end = '\0';
tabsize_set = true;
2016-08-24 22:02:38 +02:00
tabsize = r_num_math (NULL, fmt + i + 1);
2014-10-07 17:51:15 +02:00
*end = ']';
while (fmt[i++] != ']') {
;
}
2014-10-07 17:51:15 +02:00
} else {
tabsize = 1;
}
switch (fmt[i]) {
case '.':
idx--;
case 'c':
case 'b':
case 'X':
size += tabsize * 1;
break;
case 'w':
size += tabsize * 2;
break;
case ':':
idx--;
case 'd':
case 'o':
case 'i':
case 'x':
case 'f':
case 's':
case 't':
2016-08-02 00:37:07 +02:00
size += tabsize * 4;
break;
case 'S':
case 'q':
case 'F':
2016-08-02 00:37:07 +02:00
size += tabsize * 8;
break;
case 'z':
case 'Z':
size += tabsize;
break;
case '*':
size += tabsize * (p->bits / 8);
i++;
idx--; //no need to go ahead for args
break;
case 'B':
case 'E':
if (tabsize_set) {
switch (tabsize) {
case 1: size += 1; break;
case 2: size += 2; break;
case 4: size += 4; break;
case 8: size += 8; break;
default:
eprintf ("Unknown enum format size: %d\n", tabsize);
break;
}
} else {
size += 4; // Assuming by default enum as int
}
break;
case '?':
{
const char *wordAtIndex = NULL;
const char *format = NULL;
char *endname = NULL, *structname = NULL;
char tmp = 0;
if (words < idx) {
eprintf ("Index out of bounds\n");
} else {
wordAtIndex = r_str_word_get0 (args, idx);
}
if (!wordAtIndex) {
break;
}
structname = strdup (wordAtIndex);
if (*structname == '(') {
endname = (char*)r_str_rchr (structname, NULL, ')');
} else {
2014-10-01 10:48:43 +02:00
free (structname);
2014-08-07 23:59:43 +02:00
break;
}
if (endname) {
*endname = '\0';
}
format = strchr (structname, ' ');
if (format) {
tmp = *format;
while (tmp == ' ') {
format++;
tmp = *format;
}
} else {
2017-02-21 23:32:57 +01:00
format = sdb_get (p->formats, structname + 1, NULL);
if (format && !strncmp (format, f, strlen (format) - 1)) { // Avoid recursion here
return -1;
}
if (!format) { // Fetch format from types db
format = r_type_format (p->sdb_types, structname + 1);
}
}
2017-09-26 23:45:42 +02:00
if (!format) {
eprintf ("Cannot find format for struct `%s'\n", structname + 1);
return 0;
}
int newsize = r_print_format_struct_size (format, p, mode, n + 1);
if (newsize < 1) {
2017-09-26 23:45:42 +02:00
eprintf ("Cannot find size for `%s'\n", format);
return 0;
}
if (format && newsize > 0) {
size += tabsize * newsize;
}
free (structname);
}
break;
case '{':
while (fmt[i] != '}') {
if (!fmt[i]) {
2016-08-24 22:02:38 +02:00
free (o);
free (args);
return -1;
}
i++;
}
i++;
idx--;
break;
case '}':
free (o);
free (args);
return -1;
case '+':
case 'e':
idx--;
break;
case 'p':
if (fmt[i+1] == '2') {
size += tabsize * 2;
} else if (fmt[i+1] == '4') {
size += tabsize * 4;
} else if (fmt[i+1] == '8') {
size += tabsize * 8;
} else {
size += tabsize * (p->bits / 8);
break;
}
i++;
break;
case 'r':
break;
case 'n':
case 'N':
if (fmt[i+1] == '1') {
size += tabsize * 1;
} else if (fmt[i+1] == '2') {
size += tabsize * 2;
} else if (fmt[i+1] == '4') {
size += tabsize * 4;
} else if (fmt[i+1] == '8') {
size += tabsize * 8;
} else {
eprintf ("Invalid n format.\n");
2016-12-14 13:21:28 +01:00
free (o);
free (args);
return -2;
}
i++;
break;
case 'D':
case 'T':
case 'u':
//TODO complete this.
default:
//idx--; //Does this makes sense?
break;
2014-08-07 23:59:43 +02:00
}
2014-10-01 10:48:43 +02:00
idx++;
2015-02-05 01:59:32 +01:00
if (mode & R_PRINT_UNIONMODE) {
2016-09-19 02:09:42 +02:00
if (size > biggest) {
biggest = size;
}
2015-02-05 01:59:32 +01:00
size = 0;
}
2014-08-07 23:59:43 +02:00
}
free (o);
2014-10-01 10:48:43 +02:00
free (args);
return (mode & R_PRINT_UNIONMODE)? biggest : size;
2014-08-07 23:59:43 +02:00
}
static int r_print_format_struct(RPrint* p, ut64 seek, const ut8* b, int len, const char *name,
2016-10-30 23:24:20 +01:00
int slide, int mode, const char *setval, char *field, int anon) {
const char *fmt;
2016-06-01 12:13:09 +02:00
char namefmt[128];
slide++;
if ((slide % STRUCTPTR) > NESTDEPTH || (slide % STRUCTFLAG)/STRUCTPTR > NESTDEPTH) {
eprintf ("Too much nested struct, recursion too deep...\n");
2014-08-07 23:59:43 +02:00
return 0;
}
if (anon) {
fmt = name;
} else {
2017-02-21 23:32:57 +01:00
fmt = sdb_get (p->formats, name, NULL);
if (!fmt) { // Fetch struct info from types DB
fmt = r_type_format (p->sdb_types, 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;
}
if (MUSTSEE && !SEEVALUE) {
2015-02-05 02:31:02 +01:00
snprintf (namefmt, sizeof (namefmt), "%%%ds", 10+6*slide%STRUCTPTR);
if (fmt[0] == '0') {
p->cb_printf (namefmt, "union");
} else {
p->cb_printf (namefmt, "struct");
}
p->cb_printf ("<%s>\n", name);
2015-02-05 02:31:02 +01:00
}
r_print_format (p, seek, b, len, fmt, mode, setval, field);
2017-09-26 23:45:42 +02:00
return r_print_format_struct_size (fmt, p, mode, 0);
2014-08-07 23:59:43 +02:00
}
static char* get_args_offset(const char *arg) {
char *args = strchr (arg, ' ');
char *sq_bracket = strchr (arg, '[');
int max = 30;
if (args && sq_bracket) {
char *csq_bracket = strchr (arg, ']');
while (args && csq_bracket && csq_bracket > args && max--) {
args = strchr (csq_bracket, ' ');
}
}
return args;
2016-02-08 11:36:34 +01:00
}
2014-12-22 11:55:07 +01:00
#define MINUSONE ((void*)(size_t)-1)
2014-11-09 20:56:22 +01:00
#define ISSTRUCT (tmp == '?' || (tmp == '*' && *(arg+1) == '?'))
2015-07-28 18:28:33 +02:00
R_API int r_print_format(RPrint *p, ut64 seek, const ut8* b, const int len,
2015-03-03 00:34:09 +01:00
const char *formatname, int mode, const char *setval, char *ofield) {
int nargs, i, j, invalid, nexti, idx, times, otimes, endian, isptr = 0;
2016-01-07 16:29:04 +01:00
const int old_bits = p->bits;
2015-03-05 23:50:46 +01:00
char *args = NULL, *bracket, tmp, last = 0;
ut64 addr = 0, addr64 = 0, seeki = 0;
static int slide = 0, oldslide = 0;
char namefmt[32], *field = NULL;
2015-03-03 00:34:09 +01:00
const char *arg = NULL;
const char *fmt = NULL;
const char *argend;
2015-07-28 18:28:33 +02:00
int viewflags = 0;
char *oarg = NULL;
2015-07-24 17:04:33 +02:00
/* Load format from name into fmt */
2016-11-03 11:59:50 +01:00
if (!formatname) {
return 0;
}
2017-02-21 23:32:57 +01:00
fmt = sdb_get (p->formats, formatname, NULL);
2016-11-03 11:59:50 +01:00
if (!fmt) {
fmt = formatname;
}
while (*fmt && IS_WHITECHAR (*fmt)) {
fmt++;
}
2016-11-03 11:59:50 +01:00
argend = fmt + strlen (fmt);
2015-03-05 23:50:46 +01:00
arg = fmt;
2015-01-29 03:01:44 +01:00
2014-08-02 14:47:45 +02:00
nexti = nargs = i = j = 0;
2016-11-03 11:59:50 +01:00
if (len < 1) {
return 0;
}
2015-06-27 19:36:30 +02:00
// len+2 to save space for the null termination in wide strings
ut8 *buf = calloc (1, len + 2);
2016-11-03 11:59:50 +01:00
if (!buf) {
return 0;
}
memcpy (buf, b, len);
2013-02-22 21:05:08 +01:00
endian = p->big_endian;
2016-11-03 11:59:50 +01:00
if (ofield && ofield != MINUSONE) {
field = strdup (ofield);
}
/* get times */
otimes = times = atoi (arg);
if (times > 0) {
2017-02-06 00:50:54 +03:00
while (IS_DIGIT(*arg)) {
2016-11-03 11:59:50 +01:00
arg++;
}
}
2014-08-02 02:19:43 +02:00
bracket = strchr (arg,'{');
if (bracket) {
char *end = strchr (arg, '}');
if (!end) {
eprintf ("No end bracket. Try pf {ecx}b @ esi\n");
goto beach;
}
2015-10-25 03:28:29 +01:00
*end = '\0';
2016-06-01 12:13:09 +02:00
times = r_num_math (NULL, bracket + 1);
arg = end + 1;
}
2016-06-01 12:13:09 +02:00
if (*arg == '\0') {
goto beach;
}
2014-08-02 02:19:43 +02:00
/* get args */
2016-02-08 11:36:34 +01:00
args = get_args_offset (arg);
if (args) {
2016-06-01 12:13:09 +02:00
int l = 0, maxl = 0;
argend = args;
tmp = *args;
while (tmp == ' ') {
args++;
tmp = *args;
}
args = strdup (args);
nargs = r_str_word_set0_stack (args);
2016-06-01 12:13:09 +02:00
if (nargs == 0) {
R_FREE (args);
2016-06-01 12:13:09 +02:00
}
for (i = 0; i < nargs; i++) {
const char *tmp = r_str_word_get0 (args, i);
const char *nm = NULL;
int len;
nm = r_str_rchr (tmp, NULL, ')');
len = strlen (nm ? nm + 1 : tmp);
2016-06-01 12:13:09 +02:00
if (len > maxl) {
2014-08-02 02:19:43 +02:00
maxl = len;
2016-06-01 12:13:09 +02:00
}
}
l++;
2016-06-01 12:13:09 +02:00
const char *ends = " "; // XXX trailing space warning
snprintf (namefmt, sizeof (namefmt), "%%%ds :%s",
((maxl + 1) * (1 + slide)) % STRUCTPTR, ends);
}
2014-10-26 02:22:58 +02:00
#define ISPOINTED ((slide%STRUCTFLAG)/STRUCTPTR<=(oldslide%STRUCTFLAG)/STRUCTPTR)
#define ISNESTED ((slide%STRUCTPTR)<=(oldslide%STRUCTPTR))
2016-06-01 12:13:09 +02:00
if (mode == R_PRINT_JSON && slide == 0) {
p->cb_printf ("[");
}
if (mode && arg[0] == '0') {
2015-02-05 01:59:32 +01:00
mode |= R_PRINT_UNIONMODE;
arg++;
} else {
mode &= ~R_PRINT_UNIONMODE;
}
if (mode & R_PRINT_DOT) {
char *fmtname;
if (formatname && *formatname) {
if (strchr (formatname, ' ')) {
fmtname = r_str_newf ("0x%"PFMT64x, seek);
} else {
fmtname = strdup (formatname);
}
} else {
fmtname = r_str_newf ("0x%"PFMT64x, seek);
}
p->cb_printf ("digraph g { graph [ rank=same; rankdir=LR; ];\n");
p->cb_printf ("root [ rank=1; shape=record\nlabel=\"%s", fmtname);
}
2015-01-29 03:01:44 +01:00
/* go format */
i = 0;
2016-06-01 12:13:09 +02:00
if (!times) {
2014-08-02 02:19:43 +02:00
otimes = times = 1;
2016-06-01 12:13:09 +02:00
}
for (; times; times--) { // repeat N times
const char * orig = arg;
2014-10-26 02:22:58 +02:00
int first = 1;
2016-06-01 12:13:09 +02:00
if (otimes > 1) {
if (mode & R_PRINT_JSON) {
2016-11-03 11:59:50 +01:00
if (otimes > times) {
p->cb_printf (",");
}
p->cb_printf ("[{\"index\":%d,\"offset\":%d},", otimes-times, seek+i);
} else if (mode) {
p->cb_printf ("0x%08"PFMT64x" [%d] {\n", seek+i, otimes-times);
}
2014-10-26 02:22:58 +02:00
}
arg = orig;
2016-06-01 12:13:09 +02:00
for (idx = 0; i < len && arg < argend && *arg; arg++) {
2015-03-18 00:59:26 +01:00
int size = 0, elem = 0; /* size of the array, element of the array */
char *fieldname = NULL, *fmtname = NULL;
2015-02-05 01:59:32 +01:00
if (mode & R_PRINT_UNIONMODE) {
i = 0;
}
2016-06-01 12:13:09 +02:00
seeki = seek + i;
addr = 0LL;
invalid = 0;
2016-01-07 16:29:04 +01:00
p->bits = old_bits;
2014-07-16 23:11:33 +02:00
if (arg[0] == '[') {
char *end = strchr (arg,']');
2016-06-01 12:13:09 +02:00
if (!end) {
2014-07-16 23:11:33 +02:00
eprintf ("No end bracket.\n");
goto beach;
}
*end = '\0';
2016-10-31 00:43:24 +01:00
size = r_get_size (p->num, buf, endian, arg + 1);
2014-07-16 23:11:33 +02:00
arg = end + 1;
*end = ']';
2014-07-16 23:11:33 +02:00
} else {
size = -1;
}
2017-09-26 23:45:42 +02:00
int fs = r_print_format_struct_size (arg, p, 0, idx);
if (fs == -2) {
2016-12-14 13:21:28 +01:00
i = -1;
goto beach;
}
2016-10-30 23:57:25 +01:00
if (fs < 1) {
fs = 4;
}
if (i + fs - 1 < len) { // should be +7 to avoid oobread on 'q'
// Max byte number where updateAddr will look into
2016-11-03 11:59:50 +01:00
if (len - i < 7) {
updateAddr (buf + i, THRESHOLD - (len - i), endian, &addr, &addr64);
} else {
2016-11-03 11:59:50 +01:00
updateAddr (buf + i, len - i, endian, &addr, &addr64);
}
2018-04-08 22:10:46 +02:00
if (p->bits == 64) {
addr = addr64;
}
} else {
2016-12-14 13:21:28 +01:00
// eprintf ("Format strings is too big for this buffer\n");
2015-06-27 19:36:30 +02:00
goto beach;
}
2014-08-01 00:15:17 +02:00
tmp = *arg;
2014-07-30 18:01:33 +02:00
if (mode && !args) {
mode |= R_PRINT_ISFIELD;
2016-06-01 12:13:09 +02:00
}
if (mode & R_PRINT_MUSTSEE && otimes > 1) {
p->cb_printf (" ");
}
if (idx < nargs && tmp != 'e' && isptr == 0) {
char *dot = NULL, *bracket = NULL;
if (field) {
dot = strchr (field, '.');
}
if (dot) {
*dot = '\0';
}
2016-06-01 12:13:09 +02:00
free (oarg);
oarg = fieldname = strdup (r_str_word_get0 (args, idx));
if (ISSTRUCT || tmp=='E' || tmp=='B' || tmp=='r') {
if (*fieldname == '(') {
fmtname = fieldname + 1;
fieldname = (char*)r_str_rchr (fieldname, NULL, ')');
if (fieldname) {
*fieldname++ = '\0';
} else {
eprintf ("Missing closing parenthesis in format ')'\n");
goto beach;
}
} else {
eprintf ("Missing name (%s)\n", fieldname);
goto beach;
}
}
if (mode && (!args || (!field && ofield != MINUSONE)
2016-06-01 12:13:09 +02:00
|| (field && !strncmp (field, fieldname, \
strchr (field, '[')
? strchr (field, '[') - field
: strlen (field) + 1)))) {
mode |= R_PRINT_ISFIELD;
} else {
mode &= ~R_PRINT_ISFIELD;
}
2015-07-23 17:10:34 +02:00
/* There we handle specific element in array */
2016-06-01 12:13:09 +02:00
if (field && (bracket = strchr (field, '[')) && mode & R_PRINT_ISFIELD) {
char *end = strchr (field, ']');
2016-06-01 12:13:09 +02:00
if (!end) {
eprintf ("Missing closing bracket\n");
goto beach;
}
*end = '\0';
elem = r_num_math (NULL, bracket+1)+1; // +1 to handle 0 index easily
for ( ; bracket < end; bracket++) {
*bracket = '\0';
}
size += elem*ARRAYINDEX_COEF;
} else {
elem = -1;
}
if (tmp != '.' && tmp != ':') {
idx++;
if (MUSTSEE && !SEEVALUE) {
p->cb_printf (namefmt, fieldname);
}
2014-07-27 20:04:27 +02:00
}
}
feed_me_again:
switch (isptr) {
2016-01-07 16:29:04 +01:00
case PTRSEEK:
2014-08-07 23:59:43 +02:00
{
nexti = i + (p->bits/8);
i = 0;
if (tmp == '?' ) {
seeki = addr;
}
memset (buf, '\0', len);
2016-06-01 12:13:09 +02:00
if (MUSTSEE) {
p->cb_printf ("(*0x%"PFMT64x")", addr);
}
isptr = (addr)? PTRBACK: NULLPTR;
2015-01-29 03:01:44 +01: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... */
p->iob.read_at (p->iob.io, (ut64)addr, buf, len-4);
2016-06-01 12:13:09 +02:00
if ( (i + 3) < len || (i + 7) < len) {
2018-04-08 22:10:46 +02:00
// XXX this breaks pf *D
if (tmp != 'D') {
updateAddr (buf + i, len - i, endian, &addr, &addr64);
}
2016-06-01 12:13:09 +02:00
} else {
eprintf ("Likely a heap buffer overflow.\n");
2015-06-27 19:36:30 +02:00
goto beach;
}
} 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);
2016-06-01 12:13:09 +02:00
p->cb_printf ("\n");
2014-08-07 23:59:43 +02:00
goto beach;
}
}
break;
2016-01-07 16:29:04 +01:00
case PTRBACK:
// restore state after pointer seek
i = nexti;
memcpy (buf, b, len);
isptr = NOPTR;
arg--;
continue;
}
2016-06-01 12:13:09 +02:00
if (tmp == 0 && last != '*') {
break;
2016-06-01 12:13:09 +02:00
}
2014-08-02 02:19:43 +02:00
/* skip chars */
switch (tmp) {
2014-08-02 14:47:45 +02:00
case '*': // next char is a pointer
isptr = PTRSEEK;
arg++;
tmp = *arg; //last;
goto feed_me_again;
2014-08-02 02:19:43 +02:00
case '+': // toggle view flags
viewflags = !viewflags;
continue;
case 'e': // tmp swap endian
endian ^= 1;
continue;
2014-08-02 02:19:43 +02:00
case ':': // skip 4 bytes
if (size == -1) {
i += 4;
} else {
while (size--) {
i+=4;
}
}
continue;
2014-08-02 02:19:43 +02:00
case '.': // skip 1 byte
2016-06-01 12:13:09 +02:00
i += (size == -1)? 1: size;
continue;
2014-08-02 02:19:43 +02:00
case 'p': // pointer reference
2016-01-07 16:29:04 +01:00
if (*(arg+1) == '2') {
tmp = 'w';
2016-01-07 16:29:04 +01:00
arg++;
} else if (*(arg+1) == '4') {
tmp = 'x';
2016-01-07 16:29:04 +01:00
arg++;
} else if (*(arg+1) == '8') {
tmp = 'q';
2016-01-07 16:29:04 +01:00
arg++;
} else { //If pointer reference is not mentioned explicitly
switch (p->bits) {
2016-01-07 16:29:04 +01:00
case 16: tmp = 'w'; break;
case 32: tmp = 'x'; break;
default: tmp = 'q'; break;
}
2016-01-07 16:29:04 +01:00
}
break;
}
2015-03-03 00:34:09 +01:00
2015-01-29 03:01:44 +01:00
/* flags */
if (mode & R_PRINT_SEEFLAGS && isptr != NULLPTR) {
2016-04-03 01:43:16 +02:00
char *newname = NULL;
if (!fieldname) {
2018-08-20 09:52:02 +03:00
newname = fieldname = r_str_newf ("pf.%"PFMT64u, seeki);
2016-04-03 01:43:16 +02:00
}
2015-03-18 00:59:26 +01:00
if (mode & R_PRINT_UNIONMODE) {
p->cb_printf ("f %s=0x%08"PFMT64x"\n", formatname, seeki);
2015-03-18 00:59:26 +01:00
goto beach;
} else if (tmp == '?') {
p->cb_printf ("f %s.%s_", fmtname, fieldname);
2014-10-26 02:41:28 +02:00
} else if (tmp == 'E') {
p->cb_printf ("f %s=0x%08"PFMT64x"\n", fieldname, seeki);
2015-01-29 03:01:44 +01:00
} else if (slide/STRUCTFLAG>0 && idx==1) {
p->cb_printf ("%s=0x%08"PFMT64x"\n", fieldname, seeki);
} else {
p->cb_printf ("f %s=0x%08" PFMT64x "\n", fieldname, seeki);
}
2016-04-03 01:43:16 +02:00
if (newname) {
R_FREE (newname);
fieldname = NULL;
2016-04-03 01:43:16 +02:00
}
}
2015-03-03 00:34:09 +01:00
/* dot */
if (mode & R_PRINT_DOT) {
if (fieldname) {
p->cb_printf ("|{0x%"PFMT64x"|%c|%s|<%s>",
seeki, tmp, fieldname, fieldname);
} else {
p->cb_printf ("|{0x%"PFMT64x"|%c|",
seeki, tmp);
}
}
2015-01-29 03:01:44 +01:00
/* json */
if (MUSTSEEJSON && mode & R_PRINT_JSON) {
if (oldslide <= slide) {
if (first) {
first = 0;
} else {
p->cb_printf (",");
}
} else if (oldslide) {
p->cb_printf ("]},");
2014-10-26 02:22:58 +02:00
}
2016-09-19 02:09:42 +02:00
if (fieldname) {
p->cb_printf ("{\"name\":\"%s\",\"type\":\"", fieldname);
} else {
p->cb_printf ("{\"type\":\"");
}
2014-10-26 02:22:58 +02:00
if (ISSTRUCT) {
p->cb_printf ("%s", fmtname);
2014-10-26 02:22:58 +02:00
} else {
if (tmp == 'n' || tmp == 'N') {
p->cb_printf ("%c%c", tmp, *(arg+1));
} else {
p->cb_printf ("%c", tmp);
}
2014-10-26 02:22:58 +02:00
}
if (isptr) {
p->cb_printf ("*");
}
p->cb_printf ("\",\"offset\":%d,\"value\":",
isptr? (seek + nexti - (p->bits / 8)) : seek + i);
2014-10-26 02:22:58 +02:00
}
bool noline = false;
2014-07-27 20:04:27 +02:00
if (isptr == NULLPTR) {
if (MUSTSEEJSON) {
p->cb_printf ("\"NULL\"}", tmp, seek + i);
} else if (MUSTSEE) {
p->cb_printf (" NULL\n");
}
isptr = PTRBACK;
} else {
/* format chars */
// before to enter in the switch statement check buf boundaries due to updateAddr
// might go beyond its len and it's usually called in each of the following functions
2015-06-27 19:36:30 +02:00
switch (tmp) {
case 'u':
2016-09-19 02:09:42 +02:00
i += r_print_format_uleb (p, endian, mode, setval, seeki, buf, i, size);
2015-06-27 19:36:30 +02:00
break;
case 't':
r_print_format_time (p, endian, mode, setval, seeki, buf, i, size);
2016-09-19 02:09:42 +02:00
i += (size==-1) ? 4 : 4*size;
2015-06-27 19:36:30 +02:00
break;
case 'q':
r_print_format_quadword (p, endian, mode, setval, seeki, buf, i, size);
2014-09-14 15:17:45 +02:00
i += (size==-1) ? 8 : 8*size;
2015-06-27 19:36:30 +02:00
break;
case 'b':
r_print_format_byte (p, endian, mode, setval, seeki, buf, i, size);
2016-09-19 02:09:42 +02:00
i += (size==-1) ? 1 : size;
2015-06-27 19:36:30 +02:00
break;
case 'C':
2016-10-31 00:43:24 +01:00
r_print_format_decchar (p, endian, mode, setval, seeki, buf, i, size);
2016-09-19 02:09:42 +02:00
i += (size==-1) ? 1 : size;
2015-06-27 19:36:30 +02:00
break;
case 'c':
2016-10-31 00:43:24 +01:00
r_print_format_char (p, endian, mode, setval, seeki, buf, i, size);
2016-09-19 02:09:42 +02:00
i += (size==-1) ? 1 : size;
2015-06-27 19:36:30 +02:00
break;
case 'X':
2016-10-31 00:43:24 +01:00
size = r_print_format_hexpairs (p, endian, mode, setval, seeki, buf, i, size);
2015-06-27 19:36:30 +02:00
i += size;
break;
case 'T':
if (r_print_format_10bytes (p, mode,
2016-09-19 02:09:42 +02:00
setval, seeki, addr, buf) == 0) {
2015-06-27 19:36:30 +02:00
i += (size==-1) ? 4 : 4*size;
2016-09-19 02:09:42 +02:00
}
2015-06-27 19:36:30 +02:00
break;
case 'f':
r_print_format_float (p, endian, mode, setval, seeki, buf, i, size);
2015-06-27 19:36:30 +02:00
i += (size==-1) ? 4 : 4*size;
break;
case 'F':
r_print_format_double (p, endian, mode, setval, seeki, buf, i, size);
i += (size==-1) ? 8 : 8*size;
break;
2015-06-27 19:36:30 +02:00
case 'i':
2017-01-12 00:31:53 +01:00
r_print_format_int (p, endian, mode, setval, seeki, buf, i, size);
i+= (size==-1) ? 4 : 4*size;
break;
case 'd': //WHY?? help says: 0x%%08x hexadecimal value (4 bytes)
r_print_format_hex (p, endian, mode, setval, seeki, buf, i, size);
2015-06-27 19:36:30 +02:00
i+= (size==-1) ? 4 : 4*size;
break;
case 'D':
2018-04-08 22:10:46 +02:00
if (isptr) {
if (p->bits == 64) {
i += r_print_format_disasm (p, addr64, size);
} else {
i += r_print_format_disasm (p, addr, size);
}
} else {
i += r_print_format_disasm (p, seeki, size);
}
2015-06-27 19:36:30 +02:00
break;
case 'o':
r_print_format_octal (p, endian, mode, setval, seeki, buf, i, size);
i += (size==-1) ? 4 : 4 * size;
break;
case ';':
noline = true;
i -= (size==-1) ? 4 : 4 * size;
if (i < 0) {
i = 0;
}
break;
case ',':
noline = true;
i -= (size==-1) ? 1 : size;
if (i < 0) {
i = 0;
}
2015-06-27 19:36:30 +02:00
break;
case 'x':
r_print_format_hexflag (p, endian, mode, setval, seeki, buf, i, size);
2016-06-01 12:13:09 +02:00
i += (size==-1) ? 4 : 4*size;
2015-06-27 19:36:30 +02:00
break;
case 'w':
r_print_format_word (p, endian, mode, setval, seeki, buf, i, size);
2016-06-01 12:13:09 +02:00
i += (size==-1) ? 2 : 2*size;
2015-06-27 19:36:30 +02:00
break;
case 'z': // zero terminated string
r_print_format_nulltermstring (p, len, endian, mode, setval, seeki, buf, i, size);
if (size == -1) {
2015-06-27 19:36:30 +02:00
i+=strlen((char*)buf+i)+1;
} else {
while (size--) {
i++;
}
}
2015-06-27 19:36:30 +02:00
break;
case 'Z': // zero terminated wide string
r_print_format_nulltermwidestring (p, len, endian, mode, setval, seeki, buf, i, size);
if (size == -1) {
i += r_wstr_clen((char*)(buf+i))*2+2;
} else {
while (size--) {
i += 2;
}
}
2015-06-27 19:36:30 +02:00
break;
case 's':
if (r_print_format_string (p, seeki, addr64, addr, 0, mode) == 0) {
2015-06-27 19:36:30 +02:00
i += (size==-1) ? 4 : 4*size;
}
2015-06-27 19:36:30 +02:00
break;
case 'S':
if (r_print_format_string (p, seeki, addr64, addr, 1, mode) == 0) {
i += (size == -1) ? 8 : 8 * size;
}
2015-06-27 19:36:30 +02:00
break;
case 'B': // resolve bitfield
if (size >= ARRAYINDEX_COEF) {
size %= ARRAYINDEX_COEF;
}
2015-06-27 19:36:30 +02:00
r_print_format_bitfield (p, seeki, fmtname, fieldname, addr, mode, size);
i+=(size==-1)?1:size;
break;
case 'E': // resolve enum
if (size >= ARRAYINDEX_COEF) {
size %= ARRAYINDEX_COEF;
}
2015-06-27 19:36:30 +02:00
r_print_format_enum (p, seeki, fmtname, fieldname, addr, mode, size);
2016-06-01 12:13:09 +02:00
i += (size==-1)? 1: size;
2015-06-27 19:36:30 +02:00
break;
case 'r':
r_print_format_register (p, mode, fmtname, setval);
break;
2015-06-27 19:36:30 +02:00
case '?':
{
int s = 0;
char *nxtfield = NULL;
char *format = NULL;
int anon = 0;
2015-06-27 19:36:30 +02:00
if (size >= ARRAYINDEX_COEF) {
2016-06-01 12:13:09 +02:00
elem = size / ARRAYINDEX_COEF - 1;
2015-06-27 19:36:30 +02:00
size %= ARRAYINDEX_COEF;
}
if (!(mode & R_PRINT_ISFIELD)) {
nxtfield = MINUSONE;
} else if (field) {
nxtfield = strchr (ofield, '.');
}
if (nxtfield != MINUSONE && nxtfield) {
nxtfield++;
}
2016-01-07 16:29:04 +01:00
2016-06-01 12:13:09 +02:00
if (MUSTSEE) {
if (!SEEVALUE) {
p->cb_printf ("\n");
}
2016-06-01 12:13:09 +02:00
}
2015-06-27 19:36:30 +02:00
if (MUSTSEEJSON) {
if (isptr) {
p->cb_printf ("%d},", seeki);
} else {
p->cb_printf ("[");
}
}
if (mode & R_PRINT_SEEFLAGS) {
slide += STRUCTFLAG;
}
2016-08-24 22:02:38 +02:00
if (!fmtname) {
break;
}
format = strchr (fmtname, ' ');
if (format) {
anon = 1;
fmtname = format;
while (*fmtname == ' ') {
fmtname++;
}
2015-06-27 19:36:30 +02:00
}
oldslide = slide;
//slide += (isptr) ? STRUCTPTR : NESTEDSTRUCT;
slide += NESTEDSTRUCT;
2015-06-27 19:36:30 +02:00
if (size == -1) {
2016-09-02 22:27:24 +02:00
s = r_print_format_struct (p, seeki,
buf + i, len - i, fmtname, slide,
mode, setval, nxtfield, anon);
2016-09-02 22:27:24 +02:00
i += (isptr) ? (p->bits / 8) : s;
if (MUSTSEEJSON) {
if (!isptr && (!arg[1] || arg[1] == ' ')) {
2016-09-02 22:27:24 +02:00
p->cb_printf ("]}");
}
}
2015-06-27 19:36:30 +02:00
} else {
if (mode & R_PRINT_ISFIELD) {
if (!SEEVALUE) {
p->cb_printf ("[\n");
}
}
2015-06-27 19:36:30 +02:00
while (size--) {
if (mode && (elem == -1 || elem == 0)) {
2015-06-27 19:36:30 +02:00
mode |= R_PRINT_MUSTSEE;
if (elem == 0) {
elem = -2;
}
2015-06-27 19:36:30 +02:00
} else {
mode &= ~R_PRINT_MUSTSEE;
}
s = r_print_format_struct (p, seek+i,
buf+i, len-i, fmtname, slide, mode, setval, nxtfield, anon);
2015-06-27 19:36:30 +02:00
if ((MUSTSEE || MUSTSEEJSON) && size != 0 && elem == -1) {
2016-09-19 02:09:42 +02:00
if (MUSTSEEJSON) {
p->cb_printf (",");
2016-09-19 05:27:42 +02:00
} else if (MUSTSEE) {
p->cb_printf ("\n");
}
}
if (elem > -1) {
elem--;
}
i += (isptr) ? (p->bits / 8) : s;
}
if (mode & R_PRINT_ISFIELD) {
if (!SEEVALUE) {
p->cb_printf ("]\n");
2015-06-27 19:36:30 +02:00
}
}
if (MUSTSEEJSON) {
2016-09-19 02:09:42 +02:00
p->cb_printf ("]}");
}
2014-10-07 17:51:15 +02:00
}
2014-10-26 02:22:58 +02:00
oldslide = slide;
//slide -= (isptr) ? STRUCTPTR : NESTEDSTRUCT;
slide -= NESTEDSTRUCT;
2015-06-27 19:36:30 +02:00
if (mode & R_PRINT_SEEFLAGS) {
oldslide = slide;
2016-09-19 02:09:42 +02:00
slide -= STRUCTFLAG;
2015-06-27 19:36:30 +02:00
}
break;
}
case 'n':
case 'N':
{
int bytes = 0;
int sign = (tmp == 'n') ? 1 : 0;
if (arg[1] == '1') {
bytes = 1;
} else if (arg[1] == '2') {
bytes = 2;
} else if (arg[1] == '4') {
bytes = 4;
} else if (arg[1] == '8') {
bytes = 8;
} else {
invalid = 1;
break;
//or goto beach;???
}
r_print_format_num (p, endian, mode, setval, seeki, buf, i, bytes, sign, size);
i += (size == -1) ? bytes : size * bytes;
arg++;
break;
}
2015-06-27 19:36:30 +02:00
default:
/* ignore unknown chars */
invalid = 1;
break;
} //switch
}
if (mode & R_PRINT_DOT) {
p->cb_printf ("}");
}
if (viewflags && p->offname) {
const char *s = p->offname (p->user, seeki);
if (s) {
p->cb_printf ("@(%s)", s);
}
s = p->offname (p->user, addr);
if (s) {
p->cb_printf ("*(%s)", s);
}
}
if (!noline && tmp != 'D' && !invalid && !fmtname && MUSTSEE) {
p->cb_printf ("\n");
2016-06-01 12:13:09 +02:00
}
last = tmp;
// XXX: Due to the already noted issues with the above, we need to strip
// args from fmt:args the same way we strip fmt BUT only for enums as
// nested structs seem to be handled correctly above!
if (arg[0] == 'E') {
char *end_fmt = strchr (arg, ' ');
if (!end_fmt) {
goto beach;
}
char *next_args = strchr (end_fmt+1, ' ');
if (next_args) {
while (*next_args != '\0') {
*end_fmt++ = *next_args++;
}
}
*end_fmt = '\0';
}
}
2016-06-01 12:13:09 +02:00
if (otimes > 1) {
if (MUSTSEEJSON) {
p->cb_printf ("]");
} else if (mode) {
p->cb_printf ("}\n");
}
}
arg = orig;
2014-10-26 02:22:58 +02:00
oldslide = 0;
}
if (mode & R_PRINT_JSON && slide == 0) {
2016-06-01 12:13:09 +02:00
p->cb_printf("]\n");
}
if (mode & R_PRINT_DOT) {
p->cb_printf ("\"];\n}\n");
// TODO: show nested structs and field reference lines
}
beach:
2015-06-27 19:36:30 +02:00
free (oarg);
free (buf);
free (field);
free (args);
return i;
}