2009-02-05 21:08:46 +00:00
|
|
|
/* radare - LGPL - Copyright 2007-2009 pancake<nopcode.org> */
|
|
|
|
|
|
|
|
#include "r_util.h"
|
|
|
|
|
|
|
|
#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
|
2010-03-18 23:36:28 +00:00
|
|
|
r_mem_copyendian((ut8*)&ret, (ut8*)&value, 8, 0);
|
2009-02-05 21:08:46 +00:00
|
|
|
#endif
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
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
|
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 void r_num_init(struct r_num_t *num) {
|
2009-02-05 21:08:46 +00:00
|
|
|
}
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API RNum *r_num_new(RNumCallback cb, void *ptr) {
|
2010-05-20 15:40:58 +00:00
|
|
|
RNum *num;
|
|
|
|
|
|
|
|
num = R_NEW (RNum);
|
|
|
|
if (num) {
|
|
|
|
num->value = 0LL;
|
|
|
|
num->callback = cb;
|
|
|
|
num->userptr = ptr;
|
|
|
|
}
|
2009-02-05 21:08:46 +00:00
|
|
|
return num;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* old get_offset */
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API ut64 r_num_get(struct r_num_t *num, const char *str) {
|
2009-02-05 21:08:46 +00:00
|
|
|
int i, j;
|
|
|
|
char lch;
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 ret = 0LL;
|
2009-02-05 21:08:46 +00:00
|
|
|
|
2010-04-11 18:50:07 +00:00
|
|
|
for (;str[0]==' ';) str = str+1;
|
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
|
|
|
|
2009-02-05 21:08:46 +00:00
|
|
|
if (str[0]=='0' && str[1]=='x') {
|
2010-04-14 11:02:23 +00:00
|
|
|
sscanf(str, "0x%"PFMT64x"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
} else {
|
|
|
|
lch = str[strlen(str)-1];
|
2010-01-21 01:38:52 +00:00
|
|
|
switch (lch) {
|
2009-02-05 21:08:46 +00:00
|
|
|
case 'h': // hexa
|
2010-04-14 11:02:23 +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;
|
|
|
|
for(j=0,i=strlen(str)-2;i>=0;i--,j++) {
|
|
|
|
if (str[i]=='1') ret|=1<<j;
|
|
|
|
else if (str[i]!='0') break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2010-04-14 11:02:23 +00:00
|
|
|
sscanf(str, "%"PFMT64d"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
break;
|
|
|
|
case 'K': case 'k':
|
2010-04-14 11:02:23 +00:00
|
|
|
sscanf(str, "%"PFMT64d"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
ret *= 1024;
|
|
|
|
break;
|
|
|
|
case 'M': case 'm':
|
2010-04-14 11:02:23 +00:00
|
|
|
sscanf(str, "%"PFMT64d"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
ret *= 1024*1024;
|
|
|
|
break;
|
|
|
|
case 'G': case 'g':
|
2010-04-14 11:02:23 +00:00
|
|
|
sscanf(str, "%"PFMT64d"", &ret);
|
2009-02-05 21:08:46 +00:00
|
|
|
ret *= 1024*1024*1024;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
R_API static ut64 r_num_math_internal(struct r_num_t *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;
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2009-08-22 04:54:41 +00:00
|
|
|
R_API ut64 r_num_math(struct r_num_t *num, const char *str)
|
2009-02-05 21:08:46 +00:00
|
|
|
{
|
2009-07-08 11:49:55 +00:00
|
|
|
ut64 ret = 0LL;
|
2009-03-12 12:30:32 +00:00
|
|
|
char op = '+';
|
2010-02-15 21:59:26 +00:00
|
|
|
int len = strlen (str)+1;
|
|
|
|
char *p, *s = alloca (len);
|
2009-02-05 21:08:46 +00:00
|
|
|
char *group;
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
do {
|
2010-02-15 21:59:26 +00:00
|
|
|
group = strchr (p, '(');
|
2009-02-05 21:08:46 +00:00
|
|
|
if (group) {
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
group[0]='(';
|
|
|
|
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) {
|
|
|
|
p2[0]='\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));
|
2009-02-05 21:08:46 +00:00
|
|
|
p =p2+1;
|
|
|
|
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;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|