2013-01-16 11:17:14 +00:00
|
|
|
/* radare - LGPL - Copyright 2007-2013 - pancake */
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
#include "r_util.h"
|
2012-02-14 03:41:40 +00:00
|
|
|
#define R_NUM_USE_CALC 1
|
|
|
|
|
2012-11-14 02:25:32 +00:00
|
|
|
R_API ut16 r_num_ntohs (ut16 foo) {
|
|
|
|
#if LIL_ENDIAN
|
|
|
|
ut8 *p = (ut8*)&foo;
|
|
|
|
foo = p[1] | p[0]<<8;
|
|
|
|
#endif
|
|
|
|
return foo;
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
#define __htonq(x) (\
|
|
|
|
(((x) & 0xff00000000000000LL) >> 56) | \
|
|
|
|
(((x) & 0x00ff000000000000LL) >> 40) | \
|
|
|
|
(((x) & 0x0000ff0000000000LL) >> 24) | \
|
|
|
|
(((x) & 0x000000ff00000000LL) >> 8) | \
|
|
|
|
(((x) & 0x00000000ff000000LL) << 8) | \
|
|
|
|
(((x) & 0x0000000000ff0000LL) << 24) | \
|
|
|
|
(((x) & 0x000000000000ff00LL) << 40) | \
|
|
|
|
(((x) & 0x00000000000000ffLL) << 56))
|
|
|
|
|
2009-08-22 04:54:41 +00:00
|
|
|
R_API ut64 r_num_htonq(ut64 value) {
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 ret = value;
|
2009-02-05 21:08:46 +00:00
|
|
|
#if LIL_ENDIAN
|
2011-06-08 23:20:02 +00:00
|
|
|
r_mem_copyendian ((ut8*)&ret, (ut8*)&value, 8, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
#endif
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-03-25 13:54:30 +00:00
|
|
|
R_API void r_num_irand() {
|
2013-01-25 02:47:09 +00:00
|
|
|
srand (r_sys_now ());
|
2011-03-25 13:54:30 +00:00
|
|
|
}
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API int r_num_rand(int max) {
|
2009-08-22 04:54:41 +00:00
|
|
|
// TODO: add srand here for security and so on
|
2011-10-25 16:58:12 +00:00
|
|
|
if (max==0) max=1;
|
2009-08-22 05:04:29 +00:00
|
|
|
return rand()%max;
|
2009-08-22 04:54:41 +00:00
|
|
|
}
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API void r_num_minmax_swap(ut64 *a, ut64 *b) {
|
2009-02-05 21:08:46 +00:00
|
|
|
if (*a>*b) {
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 tmp = *a;
|
2009-02-05 21:08:46 +00:00
|
|
|
*a = *b;
|
|
|
|
*b = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API void r_num_minmax_swap_i(int *a, int *b) {
|
2009-02-09 11:42:54 +00:00
|
|
|
if (*a>*b) {
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 tmp = *a;
|
2009-02-09 11:42:54 +00:00
|
|
|
*a = *b;
|
|
|
|
*b = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API RNum *r_num_new(RNumCallback cb, void *ptr) {
|
2010-06-23 01:33:41 +00:00
|
|
|
RNum *num = R_NEW (RNum);
|
2012-11-20 02:59:00 +00:00
|
|
|
if (!num) return NULL;
|
|
|
|
num->value = 0LL;
|
|
|
|
num->callback = cb;
|
|
|
|
num->userptr = ptr;
|
2009-02-05 21:08:46 +00:00
|
|
|
return num;
|
|
|
|
}
|
|
|
|
|
2011-09-03 03:09:31 +00:00
|
|
|
// TODO: try to avoid the use of sscanf
|
2009-02-05 21:08:46 +00:00
|
|
|
/* old get_offset */
|
2010-09-24 14:45:56 +00:00
|
|
|
R_API ut64 r_num_get(RNum *num, const char *str) {
|
2009-02-05 21:08:46 +00:00
|
|
|
int i, j;
|
2012-02-05 00:14:09 +00:00
|
|
|
ut32 s, a;
|
2011-06-04 01:14:04 +00:00
|
|
|
char lch, len;
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 ret = 0LL;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2011-06-08 23:20:02 +00:00
|
|
|
for (; *str==' '; ) str++;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
/* resolve string with an external callback */
|
|
|
|
if (num && num->callback) {
|
2010-03-30 15:37:15 +00:00
|
|
|
int ok = 0;
|
|
|
|
ret = num->callback (num->userptr, str, &ok);
|
2009-02-05 21:08:46 +00:00
|
|
|
if (ok) return ret;
|
|
|
|
}
|
|
|
|
|
2009-02-27 01:06:37 +00:00
|
|
|
if (str[0]=='\'' && str[2]=='\'')
|
2009-07-08 11:49:55 +00:00
|
|
|
return (ut64)str[1];
|
2009-02-27 01:06:37 +00:00
|
|
|
|
2012-06-15 13:39:04 +00:00
|
|
|
len = strlen (str);
|
2012-11-16 00:34:26 +00:00
|
|
|
if (len>3 && str[4] == ':') {
|
|
|
|
if (sscanf (str, "%04x", &s)==1) if (sscanf (str+5, "%04x", &a)==1) return (ut64) ((s<<4) + a);
|
2012-12-03 03:28:05 +00:00
|
|
|
} else if (len>6 && str[6] == ':') {
|
2012-11-16 00:34:26 +00:00
|
|
|
if (sscanf (str, "0x%04x:0x%04x", &s, &a) == 2) return (ut64) ((s<<4) + a);
|
|
|
|
if (sscanf (str, "0x%04x:%04x", &s, &a) == 2) return (ut64) ((s<<4) + a);
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
if (str[0]=='0' && str[1]=='x') {
|
2011-06-08 23:20:02 +00:00
|
|
|
sscanf (str, "0x%"PFMT64x"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
} else {
|
2011-06-04 01:14:04 +00:00
|
|
|
lch = str[len>0?len-1:0];
|
2011-09-03 03:09:31 +00:00
|
|
|
if (*str=='0' && lch != 'b' && lch != 'h')
|
|
|
|
lch = 'o';
|
2010-01-21 01:38:52 +00:00
|
|
|
switch (lch) {
|
2009-02-05 21:08:46 +00:00
|
|
|
case 'h': // hexa
|
2011-06-08 23:20:02 +00:00
|
|
|
sscanf (str, "%"PFMT64x"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
break;
|
|
|
|
case 'o': // octal
|
2010-04-14 11:02:23 +00:00
|
|
|
sscanf (str, "%"PFMT64o"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
break;
|
|
|
|
case 'b': // binary
|
|
|
|
ret = 0;
|
2011-06-08 23:20:02 +00:00
|
|
|
for (j=0, i=strlen (str)-2; i>=0; i--, j++) {
|
2009-02-05 21:08:46 +00:00
|
|
|
if (str[i]=='1') ret|=1<<j;
|
|
|
|
else if (str[i]!='0') break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'K': case 'k':
|
2011-06-08 23:20:02 +00:00
|
|
|
sscanf (str, "%"PFMT64d"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
ret *= 1024;
|
|
|
|
break;
|
|
|
|
case 'M': case 'm':
|
2011-06-08 23:20:02 +00:00
|
|
|
sscanf (str, "%"PFMT64d"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
ret *= 1024*1024;
|
|
|
|
break;
|
|
|
|
case 'G': case 'g':
|
2011-06-08 23:20:02 +00:00
|
|
|
sscanf (str, "%"PFMT64d"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
ret *= 1024*1024*1024;
|
|
|
|
break;
|
2012-06-27 23:27:40 +00:00
|
|
|
default:
|
|
|
|
sscanf (str, "%"PFMT64d"", &ret);
|
|
|
|
break;
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (num != NULL)
|
|
|
|
num->value = ret;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API ut64 r_num_op(char op, ut64 a, ut64 b) {
|
2010-03-30 15:37:15 +00:00
|
|
|
switch (op) {
|
2009-02-05 21:08:46 +00:00
|
|
|
case '+': return a+b;
|
|
|
|
case '-': return a-b;
|
|
|
|
case '*': return a*b;
|
2010-03-30 15:37:15 +00:00
|
|
|
case '/': return b?a/b:0;
|
2009-02-05 21:08:46 +00:00
|
|
|
case '&': return a&b;
|
|
|
|
case '|': return a|b;
|
|
|
|
case '^': return a^b;
|
|
|
|
}
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
2012-02-14 03:41:40 +00:00
|
|
|
#if !R_NUM_USE_CALC
|
2010-09-24 14:45:56 +00:00
|
|
|
R_API static ut64 r_num_math_internal(RNum *num, char *s) {
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 ret = 0LL;
|
2009-02-05 21:08:46 +00:00
|
|
|
char *p = s;
|
2011-08-07 22:46:04 +00:00
|
|
|
int i, nop, op = 0;
|
2010-02-15 21:59:26 +00:00
|
|
|
for (i=0; s[i]; i++) {
|
|
|
|
switch (s[i]) {
|
2009-02-05 21:08:46 +00:00
|
|
|
case '+':
|
|
|
|
case '-':
|
|
|
|
case '*':
|
|
|
|
case '/':
|
|
|
|
case '&':
|
|
|
|
case '^':
|
|
|
|
case '|':
|
|
|
|
nop = s[i]; s[i] = '\0';
|
2010-02-15 21:59:26 +00:00
|
|
|
ret = r_num_op (op, ret, r_num_get (num, p));
|
2009-02-05 21:08:46 +00:00
|
|
|
op = s[i] = nop; p = s + i + 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-02-15 21:59:26 +00:00
|
|
|
return r_num_op (op, ret, r_num_get (num, p));
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2012-02-14 03:41:40 +00:00
|
|
|
#endif
|
|
|
|
|
2010-09-24 14:45:56 +00:00
|
|
|
R_API ut64 r_num_math(RNum *num, const char *str) {
|
2012-02-14 03:41:40 +00:00
|
|
|
#if R_NUM_USE_CALC
|
2012-02-14 04:26:41 +00:00
|
|
|
ut64 ret;
|
|
|
|
const char *err = NULL;
|
2012-09-22 16:35:02 +00:00
|
|
|
if (!str) return 0LL;
|
2012-12-20 10:31:38 +00:00
|
|
|
//if (!str || !*str) return 0LL;
|
2012-02-14 04:26:41 +00:00
|
|
|
ret = r_num_calc (num, str, &err);
|
2012-02-27 01:02:44 +00:00
|
|
|
if (err) eprintf ("r_num_calc error: (%s) in (%s)\n", err, str);
|
2012-02-14 04:26:41 +00:00
|
|
|
else if (num) num->value = ret;
|
2012-11-30 23:15:19 +00:00
|
|
|
if (num != NULL)
|
|
|
|
num->value = ret;
|
2012-02-14 04:26:41 +00:00
|
|
|
return ret;
|
2012-02-14 03:41:40 +00:00
|
|
|
#else
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 ret = 0LL;
|
2009-03-12 12:30:32 +00:00
|
|
|
char op = '+';
|
2011-08-07 22:46:04 +00:00
|
|
|
int len;
|
|
|
|
char *p, *s, *os;
|
2009-02-05 21:08:46 +00:00
|
|
|
char *group;
|
2011-08-07 22:46:04 +00:00
|
|
|
if (!str) return 0LL;
|
|
|
|
|
|
|
|
len = strlen (str)+1;
|
|
|
|
os = malloc (len+1);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2011-07-06 09:48:16 +00:00
|
|
|
s = os;
|
2010-02-15 21:59:26 +00:00
|
|
|
memcpy (s, str, len);
|
|
|
|
for (; *s==' '; s++);
|
2009-02-05 21:08:46 +00:00
|
|
|
p = s;
|
2010-09-24 19:25:23 +00:00
|
|
|
|
2009-02-05 21:08:46 +00:00
|
|
|
do {
|
2010-02-15 21:59:26 +00:00
|
|
|
group = strchr (p, '(');
|
2009-02-05 21:08:46 +00:00
|
|
|
if (group) {
|
2011-07-06 09:48:16 +00:00
|
|
|
group[0] = '\0';
|
2010-03-30 15:37:15 +00:00
|
|
|
ret = r_num_op (op, ret, r_num_math_internal (num, p));
|
|
|
|
for (; p<group; p+=1) {
|
|
|
|
switch (*p) {
|
2009-02-05 21:08:46 +00:00
|
|
|
case '+':
|
|
|
|
case '-':
|
|
|
|
case '*':
|
|
|
|
case '/':
|
|
|
|
case '&':
|
|
|
|
case '|':
|
|
|
|
case '^':
|
|
|
|
op = *p;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-07-06 09:48:16 +00:00
|
|
|
group[0] = '(';
|
2009-02-05 21:08:46 +00:00
|
|
|
p = group+1;
|
2010-03-30 15:37:15 +00:00
|
|
|
if (r_str_delta (p, '(', ')')<0) {
|
|
|
|
char *p2 = strchr (p, '(');
|
2009-02-05 21:08:46 +00:00
|
|
|
if (p2 != NULL) {
|
2011-08-27 18:25:37 +00:00
|
|
|
*p2 = '\0';
|
2010-03-30 15:37:15 +00:00
|
|
|
ret = r_num_op (op, ret, r_num_math_internal (num, p));
|
|
|
|
ret = r_num_op (op, ret, r_num_math (num, p2+1));
|
2011-08-27 18:25:37 +00:00
|
|
|
p = p2+1;
|
2009-02-05 21:08:46 +00:00
|
|
|
continue;
|
2010-03-30 15:37:15 +00:00
|
|
|
} else eprintf ("WTF!\n");
|
|
|
|
} else ret = r_num_op (op, ret, r_num_math_internal (num, p));
|
|
|
|
} else ret = r_num_op (op, ret, r_num_math_internal (num, p));
|
|
|
|
} while (0);
|
2009-02-05 21:08:46 +00:00
|
|
|
|
|
|
|
if (num != NULL)
|
|
|
|
num->value = ret;
|
2011-07-06 09:48:16 +00:00
|
|
|
free (os);
|
2009-02-05 21:08:46 +00:00
|
|
|
return ret;
|
2012-02-14 03:41:40 +00:00
|
|
|
#endif
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
2010-09-24 14:45:56 +00:00
|
|
|
|
|
|
|
R_API int r_num_is_float(struct r_num_t *num, const char *str) {
|
|
|
|
// TODO: also support 'f' terminated strings
|
2010-10-27 14:31:51 +00:00
|
|
|
return (strchr (str, '.') != NULL)? R_TRUE:R_FALSE;
|
2010-09-24 14:45:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
R_API double r_num_get_float(struct r_num_t *num, const char *str) {
|
|
|
|
double d = 0.0f;
|
|
|
|
sscanf (str, "%lf", &d);
|
|
|
|
return d;
|
|
|
|
}
|
2010-09-24 19:25:23 +00:00
|
|
|
|
|
|
|
R_API int r_num_to_bits (char *out, ut64 num) {
|
2011-02-25 10:45:05 +00:00
|
|
|
int size = 64, i;
|
2010-09-24 19:25:23 +00:00
|
|
|
|
2011-02-25 10:45:05 +00:00
|
|
|
if (num&0xff000000) size = 32;
|
|
|
|
else if (num&0xff0000) size = 24;
|
2010-09-24 19:25:23 +00:00
|
|
|
else if (num&0xff00) size = 16;
|
|
|
|
else if (num&0xff) size = 8;
|
2011-02-25 10:45:05 +00:00
|
|
|
if (out) {
|
2011-06-08 23:20:02 +00:00
|
|
|
for (i=0; i<size; i++)
|
|
|
|
out[size-1-i] = (num>>i&1)? '1': '0';
|
2011-08-27 18:25:37 +00:00
|
|
|
out[size] = '\0'; //Maybe not nesesary?
|
2011-02-25 10:45:05 +00:00
|
|
|
}
|
2010-09-24 19:25:23 +00:00
|
|
|
return size;
|
|
|
|
}
|
2011-06-08 23:20:02 +00:00
|
|
|
|
2012-12-22 01:37:01 +00:00
|
|
|
R_API ut64 r_num_chs (int cylinder, int head, int sector, int sectorsize) {
|
|
|
|
if (sectorsize<1) sectorsize = 512;
|
|
|
|
return cylinder * head * sector * sectorsize;
|
|
|
|
}
|