2014-08-04 01:11:56 +02:00
|
|
|
/* radare - LGPL - Copyright 2012-2014 - pancake */
|
2012-11-20 03:59:00 +01:00
|
|
|
|
|
|
|
#include <r_anal.h>
|
|
|
|
|
|
|
|
#define MINLEN 1
|
2015-12-14 14:32:18 +01:00
|
|
|
static int is_string(const ut8 *buf, int size, int *len) {
|
2012-11-20 03:59:00 +01:00
|
|
|
int i;
|
2015-10-05 14:43:17 +02:00
|
|
|
if (size < 1) return 0;
|
2015-12-14 14:32:18 +01:00
|
|
|
if (size > 3 && buf[0] && !buf[1] && buf[2] && !buf[3]) {
|
2013-02-26 22:03:02 +01:00
|
|
|
*len = 1; // XXX: TODO: Measure wide string length
|
2012-11-20 03:59:00 +01:00
|
|
|
return 2; // is wide
|
2013-02-26 22:03:02 +01:00
|
|
|
}
|
2015-10-05 14:43:17 +02:00
|
|
|
for (i = 0; i < size; i++) {
|
|
|
|
if (!buf[i] && i > MINLEN) {
|
2013-02-26 22:03:02 +01:00
|
|
|
*len = i;
|
2014-08-18 15:03:02 +02:00
|
|
|
return 1;
|
2013-02-26 22:03:02 +01:00
|
|
|
}
|
2015-12-14 14:32:18 +01:00
|
|
|
if (buf[i] == 10 || buf[i] == 13 || buf[i] == 9) {
|
2015-03-31 04:21:38 +02:00
|
|
|
continue;
|
|
|
|
}
|
2015-12-14 14:32:18 +01:00
|
|
|
if (buf[i] < 32 || buf[i] > 127) {
|
2014-02-25 00:38:18 +01:00
|
|
|
// not ascii text
|
|
|
|
return 0;
|
|
|
|
}
|
2013-02-26 22:03:02 +01:00
|
|
|
if (!IS_PRINTABLE (buf[i])) {
|
|
|
|
*len = i;
|
2012-11-20 03:59:00 +01:00
|
|
|
return 0;
|
2013-02-26 22:03:02 +01:00
|
|
|
}
|
2012-11-20 03:59:00 +01:00
|
|
|
}
|
2013-02-26 22:03:02 +01:00
|
|
|
*len = i;
|
2012-11-20 03:59:00 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
static int is_number(const ut8 *buf, int endian, int size) {
|
2012-11-20 03:59:00 +01:00
|
|
|
ut64 n = r_mem_get_num (buf, size, endian);
|
2015-12-14 14:32:18 +01:00
|
|
|
return (n < UT32_MAX)? (int)n: 0;
|
2012-11-20 03:59:00 +01:00
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
static int is_null(const ut8 *buf, int size) {
|
|
|
|
const char zero[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
|
2012-11-20 03:59:00 +01:00
|
|
|
return (!memcmp (buf, &zero, size))? 1: 0;
|
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
static int is_invalid(const ut8 *buf, int size) {
|
|
|
|
if (size < 1) return 1;
|
|
|
|
if (size > 8) size = 8;
|
2012-11-20 03:59:00 +01:00
|
|
|
return (!memcmp (buf, "\xff\xff\xff\xff\xff\xff\xff\xff", size))? 1: 0;
|
|
|
|
}
|
|
|
|
|
2014-12-02 11:49:41 +01:00
|
|
|
#define USE_IS_VALID_OFFSET 1
|
2012-11-20 03:59:00 +01:00
|
|
|
static ut64 is_pointer(RIOBind *iob, const ut8 *buf, int endian, int size) {
|
2014-02-21 01:59:32 +01:00
|
|
|
ut64 n;
|
2012-11-20 03:59:00 +01:00
|
|
|
ut8 buf2[32];
|
2014-01-04 01:59:23 +01:00
|
|
|
if (size > sizeof (buf2))
|
|
|
|
size = sizeof (buf2);
|
2014-02-21 01:59:32 +01:00
|
|
|
n = r_mem_get_num (buf, size, endian);
|
2012-11-20 03:59:00 +01:00
|
|
|
if (!n) return 1; // null pointer
|
2014-12-02 11:49:41 +01:00
|
|
|
#if USE_IS_VALID_OFFSET
|
2015-07-05 01:44:45 +02:00
|
|
|
int r = iob->is_valid_offset (iob->io, n, 0);
|
2014-12-02 11:49:41 +01:00
|
|
|
return r? n: 0LL;
|
2015-07-31 12:40:04 +02:00
|
|
|
#else
|
2014-01-04 01:59:23 +01:00
|
|
|
// optimization to ignore very low and very high pointers
|
|
|
|
// this makes disasm 5x faster, but can result in some false positives
|
2014-09-20 15:21:25 +02:00
|
|
|
// we should compare with current offset, to avoid
|
|
|
|
// short/long references. and discard invalid ones
|
2015-12-14 14:32:18 +01:00
|
|
|
if (n < 0x1000) return 0; // probably wrong
|
|
|
|
if (n > 0xffffffffffffLL) return 0; // probably wrong
|
2014-01-04 01:59:23 +01:00
|
|
|
|
2014-12-04 18:38:49 +00:00
|
|
|
if (iob->read_at (iob->io, n, buf2, size) != size) return 0;
|
2012-11-20 03:59:00 +01:00
|
|
|
return is_invalid (buf2, size)? 0: n;
|
2014-12-02 11:49:41 +01:00
|
|
|
#endif
|
2012-11-20 03:59:00 +01:00
|
|
|
}
|
|
|
|
|
2014-06-15 12:08:18 +02:00
|
|
|
static int is_bin(const ut8 *buf, int size) {
|
2012-11-20 03:59:00 +01:00
|
|
|
// TODO: add more
|
2015-12-14 14:32:18 +01:00
|
|
|
if ((size >= 4 && !memcmp (buf, "\xcf\xfa\xed\xfe", 4)) || (size >= 4 && !memcmp (buf, "\x7e"
|
|
|
|
"ELF",
|
|
|
|
4)) ||
|
|
|
|
(size >= 2 && !memcmp (buf, "MZ", 2)))
|
2012-11-20 03:59:00 +01:00
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
// TODO: add is_flag, is comment?
|
2012-11-20 03:59:00 +01:00
|
|
|
|
2012-11-20 12:17:46 +01:00
|
|
|
// XXX: optimize by removing all strlens here
|
2015-12-14 14:32:18 +01:00
|
|
|
R_API char *r_anal_data_to_string(RAnalData *d) {
|
2014-10-15 02:24:22 +02:00
|
|
|
int i, len, idx, mallocsz = 1024;
|
|
|
|
ut32 n32;
|
|
|
|
char *line;
|
|
|
|
|
|
|
|
if (!d) return NULL;
|
|
|
|
|
|
|
|
line = malloc (mallocsz);
|
2015-12-14 14:32:18 +01:00
|
|
|
snprintf (line, mallocsz, "0x%08" PFMT64x " ", d->addr);
|
2014-10-15 02:24:22 +02:00
|
|
|
n32 = (ut32)d->ptr;
|
|
|
|
len = R_MIN (d->len, 8);
|
2015-12-14 14:32:18 +01:00
|
|
|
for (i = 0, idx = strlen (line); i < len; i++) {
|
|
|
|
int msz = mallocsz - idx;
|
|
|
|
if (msz > 1) {
|
|
|
|
snprintf (line + idx, msz, "%02x", d->buf[i]);
|
2014-10-15 02:24:22 +02:00
|
|
|
idx += 2;
|
|
|
|
}
|
|
|
|
}
|
2015-12-14 14:32:18 +01:00
|
|
|
if (i > 0 && d->len > len) {
|
|
|
|
int msz = mallocsz - idx;
|
|
|
|
snprintf (line + idx, msz, "..");
|
2013-03-07 13:08:05 +01:00
|
|
|
idx += 2;
|
2014-10-15 02:24:22 +02:00
|
|
|
msz -= 2;
|
2013-03-07 13:08:05 +01:00
|
|
|
}
|
2012-11-20 12:17:46 +01:00
|
|
|
strcat (line, " ");
|
2014-06-05 01:32:05 +02:00
|
|
|
idx += 2;
|
2015-07-31 12:40:04 +02:00
|
|
|
if (mallocsz - idx > 12) {
|
|
|
|
switch (d->type) {
|
|
|
|
case R_ANAL_DATA_TYPE_STRING:
|
2015-12-14 14:32:18 +01:00
|
|
|
snprintf (line + idx, mallocsz - idx, "string \"%s\"", d->str);
|
2015-07-31 12:40:04 +02:00
|
|
|
idx = strlen (line);
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_WIDE_STRING:
|
|
|
|
strcat (line, "wide string");
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_NUMBER:
|
|
|
|
if (n32 == d->ptr) {
|
2015-12-14 14:32:18 +01:00
|
|
|
snprintf (line + idx, mallocsz - idx,
|
2015-07-31 12:40:04 +02:00
|
|
|
"number %d 0x%x", n32, n32);
|
|
|
|
} else {
|
2015-12-14 14:32:18 +01:00
|
|
|
snprintf (line + idx, mallocsz - idx,
|
|
|
|
"number %" PFMT64d " 0x%" PFMT64x,
|
2015-07-31 12:40:04 +02:00
|
|
|
d->ptr, d->ptr);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_POINTER:
|
|
|
|
strcat (line, "pointer ");
|
2015-12-14 14:32:18 +01:00
|
|
|
sprintf (line + strlen (line), " 0x%08" PFMT64x, d->ptr);
|
2015-07-31 12:40:04 +02:00
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_INVALID:
|
|
|
|
strcat (line, "invalid");
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_HEADER:
|
|
|
|
strcat (line, "header");
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_SEQUENCE:
|
|
|
|
strcat (line, "sequence");
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_PATTERN:
|
|
|
|
strcat (line, "pattern");
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_UNKNOWN:
|
|
|
|
strcat (line, "unknown");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
strcat (line, "(null)");
|
|
|
|
break;
|
|
|
|
}
|
2012-11-20 12:17:46 +01:00
|
|
|
}
|
|
|
|
return line;
|
|
|
|
}
|
2012-11-20 03:59:00 +01:00
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
R_API RAnalData *r_anal_data_new_string(ut64 addr, const char *p, int len, int type) {
|
2012-11-20 12:17:46 +01:00
|
|
|
RAnalData *ad = R_NEW0 (RAnalData);
|
2015-06-17 12:36:08 +02:00
|
|
|
if (!ad) return NULL;
|
2013-02-26 22:03:02 +01:00
|
|
|
ad->str = NULL;
|
2012-11-20 12:17:46 +01:00
|
|
|
ad->addr = addr;
|
2012-12-06 00:55:22 +01:00
|
|
|
ad->type = type;
|
2015-07-31 12:40:04 +02:00
|
|
|
if (len == 0) {
|
2013-02-26 22:03:02 +01:00
|
|
|
len = strlen (p);
|
2015-07-31 12:40:04 +02:00
|
|
|
}
|
|
|
|
|
2013-02-26 22:03:02 +01:00
|
|
|
if (type == R_ANAL_DATA_TYPE_WIDE_STRING) {
|
2012-11-20 12:17:46 +01:00
|
|
|
/* TODO: add support for wide strings */
|
|
|
|
} else {
|
2015-12-14 14:32:18 +01:00
|
|
|
ad->str = malloc (len + 1);
|
2015-06-17 12:36:08 +02:00
|
|
|
if (!ad->str) {
|
|
|
|
free (ad);
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-02-26 22:03:02 +01:00
|
|
|
memcpy (ad->str, p, len);
|
|
|
|
ad->str[len] = 0;
|
2015-12-14 14:32:18 +01:00
|
|
|
ad->buf = malloc (len + 1);
|
|
|
|
memcpy (ad->buf, ad->str, len + 1);
|
|
|
|
ad->len = len + 1; // string length + \x00
|
2012-11-20 12:17:46 +01:00
|
|
|
}
|
2014-04-22 01:44:53 +04:00
|
|
|
ad->ptr = 0L;
|
2012-11-20 12:17:46 +01:00
|
|
|
return ad;
|
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
R_API RAnalData *r_anal_data_new(ut64 addr, int type, ut64 n, const ut8 *buf, int len) {
|
2012-11-20 12:17:46 +01:00
|
|
|
RAnalData *ad = R_NEW0 (RAnalData);
|
2014-04-09 04:53:34 +02:00
|
|
|
int l = R_MIN (len, 8);
|
2015-06-17 12:36:08 +02:00
|
|
|
if (!ad) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2015-12-14 14:32:18 +01:00
|
|
|
ad->buf = (ut8 *)&(ad->sbuf);
|
2014-02-21 01:59:32 +01:00
|
|
|
memset (ad->buf, 0, 8);
|
2015-12-14 14:32:18 +01:00
|
|
|
if (l < 1) {
|
2014-04-22 02:55:17 +04:00
|
|
|
r_anal_data_free (ad);
|
2014-04-09 04:53:34 +02:00
|
|
|
return NULL;
|
2014-04-22 02:55:17 +04:00
|
|
|
}
|
2014-04-09 04:53:34 +02:00
|
|
|
if (buf) {
|
|
|
|
memcpy (ad->buf, buf, l);
|
|
|
|
}
|
2012-11-20 12:17:46 +01:00
|
|
|
ad->addr = addr;
|
|
|
|
ad->type = type;
|
|
|
|
ad->str = NULL;
|
2014-10-15 02:24:22 +02:00
|
|
|
switch (type) {
|
2015-12-14 14:32:18 +01:00
|
|
|
case R_ANAL_DATA_TYPE_PATTERN:
|
|
|
|
case R_ANAL_DATA_TYPE_SEQUENCE:
|
|
|
|
ad->len = len;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ad->len = l;
|
2014-10-15 02:24:22 +02:00
|
|
|
}
|
2012-11-20 12:17:46 +01:00
|
|
|
ad->ptr = n;
|
|
|
|
return ad;
|
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
R_API void r_anal_data_free(RAnalData *d) {
|
2014-04-22 02:55:17 +04:00
|
|
|
if (d) {
|
2015-12-14 14:32:18 +01:00
|
|
|
if (d->buf != (ut8 *)&(d->sbuf)) free (d->buf);
|
2014-04-22 02:55:17 +04:00
|
|
|
if (d->str != NULL) free (d->str);
|
2014-04-09 09:51:10 -05:00
|
|
|
free (d);
|
|
|
|
}
|
2012-11-20 12:17:46 +01:00
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
R_API RAnalData *r_anal_data(RAnal *anal, ut64 addr, const ut8 *buf, int size) {
|
2014-04-09 09:51:10 -05:00
|
|
|
ut64 dst = 0;
|
2013-02-26 22:03:02 +01:00
|
|
|
int n, nsize = 0;
|
2012-11-20 03:59:00 +01:00
|
|
|
int bits = anal->bits;
|
|
|
|
int endi = !anal->big_endian;
|
2015-12-14 14:32:18 +01:00
|
|
|
int word = R_MIN (8, bits / 8);
|
2012-11-20 03:59:00 +01:00
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
if (size < 4)
|
2014-10-15 02:24:22 +02:00
|
|
|
return NULL;
|
2014-06-15 12:08:18 +02:00
|
|
|
if (size >= word && is_invalid (buf, word))
|
2014-04-09 04:53:34 +02:00
|
|
|
return r_anal_data_new (addr, R_ANAL_DATA_TYPE_INVALID,
|
2015-12-14 14:32:18 +01:00
|
|
|
-1, buf, word);
|
2014-10-15 02:24:22 +02:00
|
|
|
{
|
|
|
|
int i, len = R_MIN (size, 64);
|
|
|
|
int is_pattern = 0;
|
|
|
|
int is_sequence = 0;
|
|
|
|
char ch = buf[0];
|
2015-12-14 14:32:18 +01:00
|
|
|
char ch2 = ch + 1;
|
|
|
|
for (i = 1; i < len; i++) {
|
2014-10-15 02:24:22 +02:00
|
|
|
if (ch2 == buf[i]) {
|
|
|
|
ch2++;
|
|
|
|
is_sequence++;
|
|
|
|
} else is_sequence = 0;
|
2015-12-14 14:32:18 +01:00
|
|
|
if (ch == buf[i]) {
|
2014-10-15 02:24:22 +02:00
|
|
|
is_pattern++;
|
|
|
|
}
|
|
|
|
}
|
2015-12-14 14:32:18 +01:00
|
|
|
if (is_sequence > len - 2) {
|
2014-10-15 02:24:22 +02:00
|
|
|
return r_anal_data_new (addr, R_ANAL_DATA_TYPE_SEQUENCE, -1,
|
2015-12-14 14:32:18 +01:00
|
|
|
buf, is_sequence);
|
2014-10-15 02:24:22 +02:00
|
|
|
}
|
2015-12-14 14:32:18 +01:00
|
|
|
if (is_pattern > len - 2) {
|
2014-10-15 02:24:22 +02:00
|
|
|
return r_anal_data_new (addr, R_ANAL_DATA_TYPE_PATTERN, -1,
|
2015-12-14 14:32:18 +01:00
|
|
|
buf, is_pattern);
|
2014-10-15 02:24:22 +02:00
|
|
|
}
|
|
|
|
}
|
2014-06-15 12:08:18 +02:00
|
|
|
if (size >= word && is_null (buf, word))
|
2014-04-09 04:53:34 +02:00
|
|
|
return r_anal_data_new (addr, R_ANAL_DATA_TYPE_NULL,
|
2015-12-14 14:32:18 +01:00
|
|
|
-1, buf, word);
|
2014-06-15 12:08:18 +02:00
|
|
|
if (is_bin (buf, size))
|
|
|
|
return r_anal_data_new (addr, R_ANAL_DATA_TYPE_HEADER, -1,
|
2015-12-14 14:32:18 +01:00
|
|
|
buf, word);
|
2015-10-05 14:43:17 +02:00
|
|
|
if (size >= word) {
|
2014-04-09 04:53:34 +02:00
|
|
|
dst = is_pointer (&anal->iob, buf, endi, word);
|
|
|
|
if (dst) return r_anal_data_new (addr,
|
2015-12-14 14:32:18 +01:00
|
|
|
R_ANAL_DATA_TYPE_POINTER, dst, buf, word);
|
2014-04-09 04:53:34 +02:00
|
|
|
}
|
2013-02-26 22:03:02 +01:00
|
|
|
switch (is_string (buf, size, &nsize)) {
|
2014-04-09 04:53:34 +02:00
|
|
|
case 1: return r_anal_data_new_string (addr, (const char *)buf,
|
2015-12-14 14:32:18 +01:00
|
|
|
nsize, R_ANAL_DATA_TYPE_STRING);
|
2014-04-09 04:53:34 +02:00
|
|
|
case 2: return r_anal_data_new_string (addr, (const char *)buf,
|
2015-12-14 14:32:18 +01:00
|
|
|
nsize, R_ANAL_DATA_TYPE_WIDE_STRING);
|
2013-02-26 22:03:02 +01:00
|
|
|
}
|
2014-06-15 12:08:18 +02:00
|
|
|
if (size >= word) {
|
|
|
|
n = is_number (buf, endi, word);
|
|
|
|
if (n) return r_anal_data_new (addr, R_ANAL_DATA_TYPE_NUMBER,
|
2015-12-14 14:32:18 +01:00
|
|
|
n, buf, word);
|
2014-06-15 12:08:18 +02:00
|
|
|
}
|
2014-04-09 04:53:34 +02:00
|
|
|
return r_anal_data_new (addr, R_ANAL_DATA_TYPE_UNKNOWN, dst,
|
2015-12-14 14:32:18 +01:00
|
|
|
buf, R_MIN (word, size));
|
2012-11-20 03:59:00 +01:00
|
|
|
}
|
|
|
|
|
2015-12-14 14:32:18 +01:00
|
|
|
R_API const char *r_anal_data_kind(RAnal *a, ut64 addr, const ut8 *buf, int len) {
|
2012-11-20 03:59:00 +01:00
|
|
|
int inv = 0;
|
|
|
|
int unk = 0;
|
|
|
|
int str = 0;
|
2012-11-20 12:17:46 +01:00
|
|
|
int num = 0;
|
2012-11-20 03:59:00 +01:00
|
|
|
int i, j;
|
2012-11-20 12:17:46 +01:00
|
|
|
RAnalData *data;
|
2015-12-14 14:32:18 +01:00
|
|
|
int word = a->bits / 8;
|
|
|
|
for (i = j = 0; i < len; j++) {
|
2014-11-07 00:41:29 +01:00
|
|
|
if (str && !buf[i])
|
2015-12-14 14:32:18 +01:00
|
|
|
str++;
|
|
|
|
data = r_anal_data (a, addr + i, buf + i, len - i);
|
2014-10-15 02:24:22 +02:00
|
|
|
if (data == NULL) {
|
2015-12-14 14:32:18 +01:00
|
|
|
i += word;
|
2014-10-15 02:24:22 +02:00
|
|
|
continue;
|
|
|
|
}
|
2012-11-20 12:17:46 +01:00
|
|
|
switch (data->type) {
|
2012-11-20 03:59:00 +01:00
|
|
|
case R_ANAL_DATA_TYPE_INVALID:
|
|
|
|
inv++;
|
|
|
|
i += word;
|
|
|
|
break;
|
2012-11-20 12:17:46 +01:00
|
|
|
case R_ANAL_DATA_TYPE_NUMBER:
|
2015-12-14 14:32:18 +01:00
|
|
|
if (data->ptr > 1000) num++;
|
2012-11-20 12:17:46 +01:00
|
|
|
i += word;
|
|
|
|
break;
|
2012-11-20 03:59:00 +01:00
|
|
|
case R_ANAL_DATA_TYPE_UNKNOWN:
|
|
|
|
unk++;
|
|
|
|
i += word;
|
|
|
|
break;
|
|
|
|
case R_ANAL_DATA_TYPE_STRING:
|
2015-12-14 14:32:18 +01:00
|
|
|
if (data->len > 0) {
|
2015-07-31 12:40:04 +02:00
|
|
|
i += data->len;
|
2015-12-14 14:32:18 +01:00
|
|
|
} else i += word;
|
2012-11-20 03:59:00 +01:00
|
|
|
str++;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
i += word;
|
|
|
|
}
|
2012-11-20 12:17:46 +01:00
|
|
|
r_anal_data_free (data);
|
2015-07-31 12:40:04 +02:00
|
|
|
}
|
2015-12-14 14:32:18 +01:00
|
|
|
if (j < 1) return "unknown";
|
|
|
|
if ((inv * 100 / j) > 60) return "invalid";
|
|
|
|
if ((unk * 100 / j) > 60) return "code";
|
|
|
|
if ((num * 100 / j) > 60) return "code";
|
|
|
|
if ((str * 100 / j) > 40) return "text";
|
2012-11-20 03:59:00 +01:00
|
|
|
return "data";
|
|
|
|
}
|