/* radare - LGPL - Copyright 2007-2018 - pancake */ #if __WINDOWS__ && MINGW32 && !__CYGWIN__ #include #endif #include /* for ceill */ #include #define R_NUM_USE_CALC 1 static ut64 r_num_tailff(RNum *num, const char *hex); // TODO: rename to r_num_srand() static void r_srand (int seed) { #if HAVE_ARC4RANDOM_UNIFORM // no-op (void)seed; #else srand (seed); #endif } static int r_rand (int mod) { #if HAVE_ARC4RANDOM_UNIFORM return (int)arc4random_uniform (mod); #else return rand ()%mod; #endif } R_API void r_num_irand() { r_srand (r_sys_now ()); } R_API int r_num_rand(int max) { static bool rand_initialized = false; if (!rand_initialized) { r_num_irand (); rand_initialized = true; } if (!max) { max = 1; } return r_rand (max); } R_API void r_num_minmax_swap(ut64 *a, ut64 *b) { if (*a > *b) { ut64 tmp = *a; *a = *b; *b = tmp; } } R_API void r_num_minmax_swap_i(int *a, int *b) { if (*a > *b) { ut64 tmp = *a; *a = *b; *b = tmp; } } R_API RNum *r_num_new(RNumCallback cb, RNumCallback2 cb2, void *ptr) { RNum *num = R_NEW0 (RNum); if (!num) { return NULL; } num->value = 0LL; num->callback = cb; num->cb_from_value = cb2; num->userptr = ptr; return num; } R_API void r_num_free(RNum *num) { R_FREE (num); } #define KB (1ULL << 10) #define MB (1ULL << 20) #define GB (1ULL << 30) #define TB (1ULL << 40) #define PB (1ULL << 50) #define EB (1ULL << 60) /** * Convert size in bytes to human-readable string * * Result is stored in buf (buf should be at least 8 bytes in size). * If buf is NULL, memory for the new string is obtained with malloc(3), * and can be freed with free(3). * * On success, returns a pointer to buf. It returns NULL if * insufficient memory was available. */ R_API char *r_num_units(char *buf, size_t len, ut64 num) { long double fnum; char unit; const char *fmt_str; if (!buf) { buf = malloc (len + 1); if (!buf) { return NULL; } } fnum = num; if (num >= EB) { unit = 'E'; fnum /= EB; } else if (num >= PB) { unit = 'P'; fnum /= PB; } else if (num >= TB) { unit = 'T'; fnum /= TB; } else if (num >= GB) { unit = 'G'; fnum /= GB; } else if (num >= MB) { unit = 'M'; fnum /= MB; } else if (num >= KB) { unit = 'K'; fnum /= KB; } else { unit = '\0'; } fmt_str = (ceill (fnum) == fnum) ? "%.0" LDBLFMT "%c" : "%.1" LDBLFMT "%c"; snprintf (buf, len, fmt_str, fnum, unit); return buf; } R_API const char *r_num_get_name(RNum *num, ut64 n) { if (num->cb_from_value) { int ok = 0; const char *msg = num->cb_from_value (num, n, &ok); if (msg && *msg) { return msg; } if (ok) { return msg; } } return NULL; } static void error(RNum *num, const char *err_str) { if (num) { num->nc.errors++; #if 0 num->nc.calc_err = err_str; #endif } } // TODO: try to avoid the use of sscanf /* old get_offset */ R_API ut64 r_num_get(RNum *num, const char *str) { int i, j, ok; char lch, len; ut64 ret = 0LL; ut32 s, a; if (num && !num->nc.under_calc) { num->nc.errors = 0; } if (!str) { return 0; } for (; *str==' '; ) { str++; } if (!*str) { return 0; } if (!strncmp (str, "1u", 2)) { // '1' is captured by op :( if (num && num->value == UT64_MAX) { num->value = 0; } switch (atoi (str + 2)) { case 64: return (ut64)UT64_MAX; case 32: return (ut64)UT32_MAX; case 16: return (ut64)UT16_MAX; case 8: return (ut64)UT8_MAX; } } /* resolve string with an external callback */ if (num && num->callback) { ok = 0; ret = num->callback (num->userptr, str, &ok); if (ok) { return ret; } } if (str[0] && str[1] && str[2]) { if (str[0]=='\'' && str[2]=='\'') { return (ut64)str[1]; } } len = strlen (str); if (len > 3 && str[4] == ':') { if (sscanf (str, "%04x", &s) == 1) { if (sscanf (str + 5, "%04x", &a)==1) { return (ut64) ((s<<4) + a); } } } else if (len>6 && str[6] == ':') { 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); } } if (str[0] == '0' && str[1] == 'b') { ret = 0; for (j = 0, i = strlen (str) - 1; i > 0; i--, j++) { if (str[i] == '1') { ret|=1 << j; } else if (str[i] != '0') { break; } } sscanf (str, "0x%"PFMT64x, &ret); } else if (str[0] == '\'') { ret = str[1] & 0xff; // ugly as hell } else if (!strncmp (str, "0xff..", 6) || !strncmp (str, "0xFF..", 6)) { ret = r_num_tailff (num, str + 5); // ugly as hell } else if (!strncmp (str, "0xf..", 5) || !strncmp (str, "0xF..", 5)) { ret = r_num_tailff (num, str + 5); } else if (str[0] == '0' && str[1] == 'x') { const char *lodash = strchr (str + 2, '_'); if (lodash) { // Support 0x1000_f000_4000 // TODO: Only take underscores separated every 4 chars starting at the end char *s = strdup (str + 2); r_str_replace_char (s, '_', 0); ret = strtoull (s, NULL, 16); free (s); } else { ret = strtoull (str + 2, NULL, 16); // sscanf (str+2, "%"PFMT64x, &ret); } } else { char *endptr; int len_num = len > 0 ? len - 1 : 0; int chars_read = len_num; bool zero_read = false; lch = str[len > 0 ? len - 1 : 0]; if (*str == '0' && IS_DIGIT (*(str + 1)) && lch != 'b' && lch != 'h') { lch = 'o'; len_num++; } switch (lch) { case 'h': // hexa if (!sscanf (str, "%"PFMT64x"%n", &ret, &chars_read) || chars_read != len_num) { error (num, "invalid hex number"); } break; case 'o': // octal if (!sscanf (str, "%"PFMT64o"%n", &ret, &chars_read) || chars_read != len_num) { error (num, "invalid octal number"); } break; case 'b': // binary ret = 0; ok = true; for (j = 0, i = strlen (str) - 2; i >= 0; i--, j++) { if (str[i] == '1') { ret|=1 << j; } else if (str[i] != '0') { ok = false; break; } } if (!ok || !len_num) { error (num, "invalid binary number"); } break; case 't': // ternary ret = 0; ok = true; ut64 x = 1; for (i = strlen (str) - 2; i >= 0; i--) { if (str[i] < '0' || '2' < str[i]) { ok = false; break; } ret += x * (str[i] - '0'); x *= 3; } if (!ok || !len_num) { error (num, "invalid ternary number"); } break; case 'K': case 'k': if (strchr (str, '.')) { double d = 0; if (sscanf (str, "%lf%n", &d, &chars_read)) { ret = (ut64)(d * KB); } else { zero_read = true; } } else { if (sscanf (str, "%"PFMT64d"%n", &ret, &chars_read)) { ret *= KB; } else { zero_read = true; } } if (zero_read || chars_read != len_num) { error (num, "invalid kilobyte number"); } break; case 'M': case 'm': if (strchr (str, '.')) { double d = 0; if (sscanf (str, "%lf%n", &d, &chars_read)) { ret = (ut64)(d * MB); } else { zero_read = true; } } else { if (sscanf (str, "%"PFMT64d"%n", &ret, &chars_read)) { ret *= MB; } else { zero_read = true; } } if (zero_read || chars_read != len_num) { error (num, "invalid megabyte number"); } break; case 'G': case 'g': if (strchr (str, '.')) { double d = 0; if (sscanf (str, "%lf%n", &d, &chars_read)) { ret = (ut64)(d * GB); } else { zero_read = true; } } else { if (sscanf (str, "%"PFMT64d"%n", &ret, &chars_read)) { ret *= GB; } else { zero_read = true; } } if (zero_read || chars_read != len_num) { error (num, "invalid gigabyte number"); } break; default: #if 0 //sscanf (str, "%"PFMT64d"%n", &ret, &chars_read); // 32bit chop #if __WINDOWS__ && MINGW32 && !__CYGWIN__ ret = _strtoui64 (str, &endptr, 10); #endif #endif ret = strtoull (str, &endptr, 10); if (!IS_DIGIT (*str) || (*endptr && *endptr != lch)) { error (num, "unknown symbol"); } break; } } if (num) { num->value = ret; } return ret; } #if !R_NUM_USE_CALC static ut64 r_num_op(RNum *num, char op, ut64 a, ut64 b) { switch (op) { case '+': return a+b; case '-': return a-b; case '*': return a*b; case '/': if (!b && num) num->dbz = 1; return b?a/b:0; case '&': return a&b; case '|': return a|b; case '^': return a^b; } return b; } R_API static ut64 r_num_math_internal(RNum *num, char *s) { ut64 ret = 0LL; char *p = s; int i, nop, op = 0; for (i=0; s[i]; i++) { if (r_num_is_op (s[i])) { nop = s[i]; s[i] = '\0'; ret = r_num_op (num, op, ret, r_num_get (num, p)); op = s[i] = nop; p = s + i + 1; break; } } return r_num_op (op, ret, r_num_get (num, p)); } #endif /* !R_NUM_USE_CALC */ R_API ut64 r_num_math(RNum *num, const char *str) { #if R_NUM_USE_CALC ut64 ret; const char *err = NULL; if (!str) { return 0LL; } //if (!str || !*str) return 0LL; if (num) { num->dbz = 0; } ret = r_num_calc (num, str, &err); if (err) { eprintf ("r_num_calc error: (%s) in (%s)\n", err, str); } if (num) { num->value = ret; } return ret; #else ut64 ret = 0LL; char op = '+'; int len; char *p, *s, *os; char *group; if (!str) return 0LL; len = strlen (str)+1; os = malloc (len+1); s = os; memcpy (s, str, len); for (; *s==' '; s++); p = s; do { group = strchr (p, '('); if (group) { group[0] = '\0'; ret = r_num_op (op, ret, r_num_math_internal (num, p)); for (; pvalue = ret; } free (os); return ret; #endif } R_API int r_num_is_float(RNum *num, const char *str) { return (IS_DIGIT (*str) && (strchr (str, '.') || str[strlen (str) - 1] == 'f')); } R_API double r_num_get_float(RNum *num, const char *str) { double d = 0.0f; (void) sscanf (str, "%lf", &d); return d; } R_API int r_num_to_bits (char *out, ut64 num) { int size = 64, i; if (num >> 32) { size = 64; } else if (num & 0xff000000) { size = 32; } else if (num & 0xff0000) { size = 24; } else if (num & 0xff00) { size = 16; } else if (num & 0xff) { size = 8; } if (out) { int pos = 0; int realsize = 0; int hasbit = 0; for (i=0; i>(size-i-1))&1)? '1': '0'; if (hasbit || bit=='1') { out[pos++] = bit;//size-1-i] = bit; } if (!hasbit && bit=='1') { hasbit=1; realsize = size-i; } } if (realsize == 0) { out[realsize++] = '0'; } out[realsize] = '\0'; //Maybe not nesesary? } return size; } R_API int r_num_to_trits (char *out, ut64 num) { if (out == NULL) { return false; } int i; for (i = 0; num; i++, num /= 3) { out[i] = (char) ('0' + num % 3); } if (i == 0) { out[0] = '0'; i++; } out[i] = '\0'; r_str_reverse (out); return true; } R_API ut64 r_num_chs (int cylinder, int head, int sector, int sectorsize) { if (sectorsize < 1) { sectorsize = 512; } return (ut64)cylinder * (ut64)head * (ut64)sector * (ut64)sectorsize; } R_API int r_num_conditional(RNum *num, const char *str) { char *lgt, *t, *p, *s = strdup (str); int res = 0; ut64 n, a, b; p = s; do { t = strchr (p, ','); if (t) { *t = 0; } lgt = strchr (p, '<'); if (lgt) { *lgt = 0; a = r_num_math (num, p); if (lgt[1]=='=') { b = r_num_math (num, lgt+2); if (a > b) { goto fail; } } else { b = r_num_math (num, lgt+1); if (a >= b) { goto fail; } } } else { lgt = strchr (p, '>'); if (lgt) { *lgt = 0; a = r_num_math (num, p); if (lgt[1]=='=') { b = r_num_math (num, lgt+2); if (a < b) { goto fail; } } else { b = r_num_math (num, lgt+1); if (a <= b) { goto fail; } } } else { lgt = strchr (p, '='); if (lgt && lgt > p) { lgt--; if (*lgt == '!') { r_str_replace_char (p, '!', ' '); r_str_replace_char (p, '=', '-'); n = r_num_math (num, p); if (!n) { goto fail; } } } lgt = strstr (p, "=="); if (lgt) { *lgt = ' '; } r_str_replace_char (p, '=', '-'); n = r_num_math (num, p); if (n) { goto fail; } } } p = t+1; } while (t); res = 1; fail: free (s); return res; } R_API int r_num_is_valid_input(RNum *num, const char *input_value) { ut64 value = input_value ? r_num_math (num, input_value) : 0; return !(value == 0 && input_value && *input_value != '0') || !(value == 0 && input_value && *input_value != '@'); } R_API ut64 r_num_get_input_value(RNum *num, const char *input_value) { ut64 value = input_value ? r_num_math (num, input_value) : 0; return value; } #define NIBBLE_TO_HEX(n) (((n) & 0xf) > 9 ? 'a' + ((n) & 0xf) - 10 : '0' + ((n) & 0xf)) static int escape_char(char* dst, char byte) { const char escape_map[] = "abtnvfr"; if (byte >= 7 && byte <= 13) { *(dst++) = '\\'; *(dst++) = escape_map [byte - 7]; *dst = 0; return 2; } else if (byte) { *(dst++) = '\\'; *(dst++) = 'x'; *(dst++) = NIBBLE_TO_HEX (byte >> 4); *(dst++) = NIBBLE_TO_HEX (byte); *dst = 0; return 4; } return 0; } R_API char* r_num_as_string(RNum *___, ut64 n, bool printable_only) { char str[34]; // 8 byte * 4 chars in \x?? format int stri, ret = 0, off = 0; int len = sizeof (ut64); ut64 num = n; str[stri=0] = 0; while (len--) { char ch = (num & 0xff); if (ch >= 32 && ch < 127) { str[stri++] = ch; str[stri] = 0; } else if (!printable_only && (off = escape_char (str + stri, ch)) != 0) { stri += off; } else { if (ch) { return NULL; } } ret |= (num & 0xff); num >>= 8; } if (ret) { return strdup (str); } else if (!printable_only) { return strdup ("\\0"); } return NULL; } R_API int r_is_valid_input_num_value(RNum *num, const char *input_value){ ut64 value = input_value ? r_num_math (num, input_value) : 0; return !(value == 0 && input_value && *input_value == '0'); } R_API ut64 r_get_input_num_value(RNum *num, const char *input_value){ ut64 value = input_value ? r_num_math (num, input_value) : 0; return value; } static inline int get_nth_nibble (ut64 n, int i) { int sz = (sizeof (n) << 1) - 1; int s = (sz - i) * 4; return (n >> s) & 0xf; } R_API ut64 r_num_tail_base(RNum *num, ut64 addr, ut64 off) { int i; bool ready = false; ut64 res = 0; for (i = 0; i < 16; i++) { ut64 o = get_nth_nibble (off, i); if (!ready) { bool iseq = get_nth_nibble (addr, i) == o; if (i == 0 && !iseq) { return UT64_MAX; } if (iseq) { continue; } } ready = true; ut8 pos = (15 - i) * 4; res |= (o << pos); } return res; } R_API ut64 r_num_tail(RNum *num, ut64 addr, const char *hex) { ut64 mask = 0LL; ut64 n = 0; char *p; int i; while (*hex && (*hex == ' ' || *hex=='.')) { hex++; } i = strlen (hex) * 4; p = malloc (strlen (hex) + 10); if (p) { strcpy (p, "0x"); strcpy (p + 2, hex); if (isxdigit ((ut8)hex[0])) { n = r_num_math (num, p); } else { eprintf ("Invalid argument\n"); free (p); return addr; } free (p); } mask = UT64_MAX << i; return (addr & mask) | n; } static ut64 r_num_tailff(RNum *num, const char *hex) { ut64 n = 0; char *p; int i; while (*hex && (*hex == ' ' || *hex=='.')) { hex++; } i = strlen (hex) * 4; p = malloc (strlen (hex) + 10); if (p) { strcpy (p, "0x"); strcpy (p + 2, hex); if (isxdigit ((ut8)hex[0])) { n = r_num_math (num, p); } else { eprintf ("Invalid argument\n"); free (p); return UT64_MAX; } free (p); } ut64 left = ((UT64_MAX >>i) << i); return left | n; } R_API int r_num_between(RNum *num, const char *input_value) { int i; ut64 ns[3]; char * const str = strdup (input_value); RList *nums = r_num_str_split_list (str); int len = r_list_length (nums); if (len < 3) { free (str); r_list_free (nums); return -1; } if (len > 3) { len = 3; } for (i = 0; i < len; i++) { ns[i] = r_num_math (num, r_list_pop_head (nums)); } free (str); r_list_free (nums); return num->value = R_BETWEEN (ns[0], ns[1], ns[2]); } R_API bool r_num_is_op(const char c) { return c == '/' || c == '+' || c == '-' || c == '*' || c == '%' || c == '&' || c == '^' || c == '|'; } //Assumed *str is parsed as an expression correctly R_API int r_num_str_len (const char *str) { int i = 0, len = 0, st; st = 0;//0: number, 1: op if (str[0] == '(') { i++; } while (str[i] != '\0') { switch (st) { case 0: //number while (!r_num_is_op (str[i]) && str[i] != ' ' && str[i] != '\0') { i++; if (str[i] == '(') { i += r_num_str_len (str+i); } } len = i; st = 1; break; case 1: //op while (str[i] != '\0' && str[i] == ' ') { i++; } if (!r_num_is_op (str[i])) { return len; } if (str[i] == ')') { return i + 1; } i++; while (str[i] != '\0' && str[i] == ' ') { i++; } st = 0; break; } } return len; } R_API int r_num_str_split(char *str) { int i = 0, count = 0; const int len = strlen (str); while (i < len) { i += r_num_str_len (str + i); str[i] = '\0'; i++; count++; } return count; } R_API RList *r_num_str_split_list(char *str) { int i, count = r_num_str_split (str); RList *list = r_list_new (); for (i = 0; i < count; i++) { r_list_append (list, str); str += strlen (str) + 1; } return list; } R_API void *r_num_dup(ut64 n) { ut64 *hn = malloc (sizeof (ut64)); if (!hn) { return NULL; } *hn = n; return (void*)hn; }