radare2/libr/util/calc.c

440 lines
10 KiB
C
Raw Normal View History

2017-11-09 11:57:51 +00:00
/* ported to C by pancake for r2 in 2012-2017 */
// TODO: integrate floating point support
// TODO: do not use global variables
/*
Reference Chapter 6:
"The C++ Programming Language", Special Edition.
2014-04-25 00:37:18 +00:00
Bjarne Stroustrup,Addison-Wesley Pub Co; 3 edition (February 15, 2000)
ISBN: 0201700735
*/
#include <r_types.h>
#include <r_util.h>
#include <ctype.h>
#include <stdio.h>
#include <unistd.h>
/* accessors */
static inline RNumCalcValue Nset(ut64 v) { RNumCalcValue n; n.d = (double)v; n.n = v; return n; }
static inline RNumCalcValue Nsetf(double v) { RNumCalcValue n; n.d = v; n.n = (ut64)v; return n; }
2014-05-02 23:23:12 +00:00
//UNUSED static inline RNumCalcValue Naddf(RNumCalcValue n, double v) { n.d += v; n.n += (ut64)v; return n; }
static inline RNumCalcValue Naddi(RNumCalcValue n, ut64 v) { n.d += (double)v; n.n += v; return n; }
static inline RNumCalcValue Nsubi(RNumCalcValue n, ut64 v) { n.d -= (double)v; n.n -= v; return n; }
static inline RNumCalcValue Nneg(RNumCalcValue n) { n.n = ~n.n; return n; }
static inline RNumCalcValue Norr(RNumCalcValue n, RNumCalcValue v) { n.d = v.d; n.n |= v.n; return n; }
static inline RNumCalcValue Nxor(RNumCalcValue n, RNumCalcValue v) { n.d = v.d; n.n ^= v.n; return n; }
static inline RNumCalcValue Nand(RNumCalcValue n, RNumCalcValue v) { n.d = v.d; n.n &= v.n; return n; }
static inline RNumCalcValue Nadd(RNumCalcValue n, RNumCalcValue v) { n.d += v.d; n.n += v.n; return n; }
static inline RNumCalcValue Nsub(RNumCalcValue n, RNumCalcValue v) { n.d -= v.d; n.n -= v.n; return n; }
static inline RNumCalcValue Nmul(RNumCalcValue n, RNumCalcValue v) {
n.d *= v.d;
n.n *= v.n;
return n;
}
2017-11-09 11:57:51 +00:00
2016-05-10 22:28:42 +00:00
static inline RNumCalcValue Nshl(RNumCalcValue n, RNumCalcValue v) { n.d += v.d; n.n <<= v.n; return n; }
static inline RNumCalcValue Nshr(RNumCalcValue n, RNumCalcValue v) { n.d += v.d; n.n >>= v.n; return n; }
2018-04-06 19:31:03 +00:00
static inline RNumCalcValue Nrol(RNumCalcValue n, RNumCalcValue v) {
n.d += v.d;
n.n = (n.n << v.n) | (n.n >> (sizeof(n.n) * 8 - v.n));
return n;
}
static inline RNumCalcValue Nror(RNumCalcValue n, RNumCalcValue v) {
n.d += v.d;
n.n = (n.n >> v.n) | (n.n << (sizeof(n.n) * 8 - v.n));
return n;
}
2014-02-26 18:42:56 +00:00
static inline RNumCalcValue Nmod(RNumCalcValue n, RNumCalcValue v) {
if (v.d) {
n.d = (n.d - (n.d / v.d));
} else {
n.d = 0;
}
if (v.n) {
n.n %= v.n;
} else {
n.n = 0;
}
2014-02-26 18:42:56 +00:00
return n;
}
2017-11-09 11:57:51 +00:00
static inline RNumCalcValue Ndiv(RNumCalcValue n, RNumCalcValue v) {
if (v.d) {
n.d /= v.d;
} else {
n.d = 0;
}
if (v.n) {
n.n /= v.n;
} else {
n.n = 0;
}
return n;
}
static RNumCalcValue expr(RNum*, RNumCalc*, int);
static RNumCalcValue term(RNum*, RNumCalc*, int);
static void error(RNum*, RNumCalc*, const char *);
static RNumCalcValue prim(RNum*, RNumCalc*, int);
static RNumCalcToken get_token(RNum*, RNumCalc*);
static void error(RNum *num, RNumCalc *nc, const char *s) {
nc->errors++;
nc->calc_err = s;
//fprintf (stderr, "error: %s\n", s);
}
static RNumCalcValue expr(RNum *num, RNumCalc *nc, int get) {
RNumCalcValue left = term (num, nc, get);
for (;;) {
switch (nc->curr_tok) {
2016-05-10 22:28:42 +00:00
case RNCSHL: left = Nshl (left, term (num, nc, 1)); break;
case RNCSHR: left = Nshr (left, term (num, nc, 1)); break;
2018-04-06 19:31:03 +00:00
case RNCROL: left = Nrol (left, term (num, nc, 1)); break;
case RNCROR: left = Nror (left, term (num, nc, 1)); break;
case RNCPLUS: left = Nadd (left, term (num, nc, 1)); break;
case RNCMINUS: left = Nsub (left, term (num, nc, 1)); break;
case RNCXOR: left = Nxor (left, term (num, nc, 1)); break;
case RNCORR: left = Norr (left, term (num, nc, 1)); break;
case RNCAND: left = Nand (left, term (num, nc, 1)); break;
default:
return left;
}
}
2013-04-02 10:45:16 +00:00
return left;
}
static RNumCalcValue term(RNum *num, RNumCalc *nc, int get) {
RNumCalcValue left = prim (num, nc, get);
for (;;) {
if (nc->curr_tok == RNCMUL) {
left = Nmul (left, prim (num, nc, 1));
2017-02-01 22:20:20 +00:00
} else if (nc->curr_tok == RNCMOD) {
2014-02-26 18:42:56 +00:00
RNumCalcValue d = prim (num, nc, 1);
if (!d.d) {
//error (num, nc, "divide by 0");
return d;
}
left = Nmod (left, d);
2017-02-01 22:20:20 +00:00
} else if (nc->curr_tok == RNCDIV) {
RNumCalcValue d = prim (num, nc, 1);
2014-11-18 00:33:56 +00:00
if (num != NULL && (!d.d || !d.n)) {
num->dbz = 1;
return d;
}
left = Ndiv (left, d);
2017-02-01 22:20:20 +00:00
} else {
return left;
}
}
}
static RNumCalcValue prim(RNum *num, RNumCalc *nc, int get) {
RNumCalcValue v = {0};
if (get) {
get_token (num, nc);
}
switch (nc->curr_tok) {
case RNCNUMBER:
v = nc->number_value;
get_token (num, nc);
return v;
case RNCNAME:
//fprintf (stderr, "error: unknown keyword (%s)\n", nc->string_value);
//double& v = table[nc->string_value];
2018-01-08 02:22:26 +00:00
r_str_trim (nc->string_value);
v = Nset (r_num_get (num, nc->string_value));
get_token (num, nc);
2014-04-25 00:37:18 +00:00
if (nc->curr_tok == RNCASSIGN) {
v = expr (num, nc, 1);
2014-04-25 00:37:18 +00:00
}
2017-02-01 22:20:20 +00:00
if (nc->curr_tok == RNCINC) {
Naddi (v, 1);
}
if (nc->curr_tok == RNCDEC) {
Nsubi (v, 1);
}
return v;
case RNCNEG:
v = nc->number_value;
get_token (num, nc);
return Nneg (nc->number_value); //prim (num, nc, 1), 1);
case RNCINC:
return Naddi (prim (num, nc, 1), 1);
case RNCDEC:
return Naddi (prim (num, nc, 1), -1);
case RNCORR:
return Norr (v, prim (num, nc, 1));
case RNCMINUS:
return Nsub (v, prim (num, nc, 1));
case RNCLEFTP:
v = expr (num, nc, 1);
2014-04-25 00:37:18 +00:00
if (nc->curr_tok == RNCRIGHTP) {
get_token (num, nc);
2017-02-01 22:20:20 +00:00
} else {
error (num, nc, " ')' expected");
}
case RNCEND:
2014-02-26 18:42:56 +00:00
case RNCXOR:
case RNCAND:
case RNCPLUS:
2014-02-26 18:42:56 +00:00
case RNCMOD:
case RNCMUL:
case RNCDIV:
case RNCPRINT:
case RNCASSIGN:
case RNCRIGHTP:
2016-05-10 22:28:42 +00:00
case RNCSHL:
case RNCSHR:
2018-04-06 19:31:03 +00:00
case RNCROL:
case RNCROR:
return v;
//default: error (num, nc, "primary expected");
}
return v;
}
static void cin_putback (RNum *num, RNumCalc *nc, char c) {
nc->oc = c;
}
R_API const char *r_num_calc_index(RNum *num, const char *p) {
if (!num) {
return NULL;
}
if (p) {
num->nc.calc_buf = p;
num->nc.calc_len = strlen (p);
num->nc.calc_i = 0;
}
return num->nc.calc_buf + num->nc.calc_i;
}
static int cin_get(RNum *num, RNumCalc *nc, char *c) {
if (nc->oc) {
*c = nc->oc;
nc->oc = 0;
} else {
2017-02-01 22:20:20 +00:00
if (!nc->calc_buf) {
return 0;
2017-02-01 22:20:20 +00:00
}
*c = nc->calc_buf[nc->calc_i];
2017-02-01 22:20:20 +00:00
if (*c) {
nc->calc_i++;
} else {
return 0;
}
}
return 1;
}
static int cin_get_num(RNum *num, RNumCalc *nc, RNumCalcValue *n) {
double d;
char str[R_NUMCALC_STRSZ]; // TODO: move into the heap?
int i = 0;
char c;
str[0] = 0;
while (cin_get (num, nc, &c)) {
2017-07-10 10:31:50 +00:00
if (c != '_' && c!=':' && c!='.' && !isalnum ((ut8)c)) {
cin_putback (num, nc, c);
break;
}
if (i < R_NUMCALC_STRSZ) {
str[i++] = c;
2014-04-25 00:37:18 +00:00
}
}
str[i] = 0;
*n = Nset (r_num_get (num, str));
if (IS_DIGIT (*str) && strchr (str, '.')) {
2017-02-01 22:20:20 +00:00
if (sscanf (str, "%lf", &d) < 1) {
return 0;
2017-02-01 22:20:20 +00:00
}
if (n->n < d) {
*n = Nsetf (d);
}
n->d = d;
}
return 1;
}
static RNumCalcToken get_token(RNum *num, RNumCalc *nc) {
2013-01-04 13:51:21 +00:00
char ch = 0, c = 0;
2017-07-10 10:31:50 +00:00
do {
if (!cin_get (num, nc, &ch)) {
return nc->curr_tok = RNCEND;
}
} while (ch != '\n' && isspace ((ut8)ch));
switch (ch) {
case 0:
case ';':
case '\n':
return nc->curr_tok = RNCEND;
case '+': // added for ++name and name++
2017-02-01 22:20:20 +00:00
if (cin_get (num, nc, &c) && c == '+') {
return nc->curr_tok = RNCINC;
2017-02-01 22:20:20 +00:00
}
cin_putback (num, nc, c);
return nc->curr_tok = (RNumCalcToken) ch;
// negate hack
case '~':
2017-02-01 22:20:20 +00:00
if (cin_get (num, nc, &c) && c == '-') {
return nc->curr_tok = RNCNEG;
2017-02-01 22:20:20 +00:00
}
cin_putback (num, nc, c);
return nc->curr_tok = (RNumCalcToken) ch;
// negative number
case '-':
2017-02-01 22:20:20 +00:00
if (cin_get (num, nc, &c) && c == '-') {
return nc->curr_tok = RNCDEC;
2017-02-01 22:20:20 +00:00
}
cin_putback (num, nc, c);
return nc->curr_tok = (RNumCalcToken) ch;
2018-04-06 19:31:03 +00:00
case '<':
if (cin_get (num, nc, &c) && c == '<') {
if (cin_get (num, nc, &c) && c == '<') {
return nc->curr_tok = RNCROL;
}
cin_putback (num, nc, c);
return nc->curr_tok = RNCSHL;
}
cin_putback (num, nc, c);
return nc->curr_tok = RNCEND;
case '>':
if (cin_get (num, nc, &c) && c == '>') {
if (cin_get (num, nc, &c) && c == '>') {
return nc->curr_tok = RNCROR;
}
cin_putback (num, nc, c);
return nc->curr_tok = RNCSHR;
}
cin_putback (num, nc, c);
return nc->curr_tok = RNCEND;
case '^':
case '&':
case '|':
case '*':
2014-02-26 18:42:56 +00:00
case '%':
case '/':
case '(':
case ')':
case '=':
return nc->curr_tok = (RNumCalcToken) ch;
case '0': case '1': case '2': case '3': case '4':
case '5': case '6': case '7': case '8': case '9':
case '.':
cin_putback (num, nc, ch);
if (!cin_get_num (num, nc, &nc->number_value)) {
error (num, nc, "invalid number conversion");
return 1;
}
return nc->curr_tok = RNCNUMBER;
2014-04-25 00:37:18 +00:00
#define isvalidchar(x) \
(isalnum(x) || (x)==':' || (x)=='$' || (x)=='.' || (x)=='_' || (x)=='?' || (x)=='\\' \
|| (x)==' ' || (x)=='[' || (x)==']' || (x)=='}' || (x)=='{' || ((x)>='0'&&(x)<='9'))
2014-04-25 00:37:18 +00:00
default:
{
int i = 0;
2016-06-14 00:19:20 +00:00
#define stringValueAppend(x) { \
const size_t max = sizeof (nc->string_value) - 1; \
if (i < max) nc->string_value[i++] = x; \
else nc->string_value[max] = 0; \
2016-06-14 00:19:20 +00:00
}
stringValueAppend(ch);
if (ch == '[') {
while (cin_get (num, nc, &ch) && ch!=']') {
2015-07-08 20:52:06 +00:00
if (i > R_NUMCALC_STRSZ - 1) {
error (num, nc, "string too long");
return 0;
}
2016-06-14 00:19:20 +00:00
stringValueAppend(ch);
}
2016-06-14 00:19:20 +00:00
stringValueAppend(ch);
} else {
while (cin_get (num, nc, &ch) && isvalidchar ((unsigned char)ch)) {
2017-02-01 22:20:20 +00:00
if (i >= R_NUMCALC_STRSZ) {
error (num, nc, "string too long");
return 0;
}
2016-06-14 00:19:20 +00:00
stringValueAppend(ch);
}
}
stringValueAppend(0);
2016-06-14 00:19:20 +00:00
if (ch!='\'') {
cin_putback (num, nc, ch);
2016-06-14 00:19:20 +00:00
}
return nc->curr_tok = RNCNAME;
}
}
}
static void load_token(RNum *num, RNumCalc *nc, const char *s) {
nc->calc_i = 0;
nc->calc_len = strlen (s);
nc->calc_buf = s;
nc->calc_err = NULL;
}
2016-05-23 20:07:32 +00:00
R_API ut64 r_num_calc(RNum *num, const char *str, const char **err) {
RNumCalcValue n;
RNumCalc *nc, nc_local;
2017-02-01 22:20:20 +00:00
if (!str || !*str) {
return 0LL;
2017-02-01 22:20:20 +00:00
}
if (num) {
nc = &num->nc;
num->dbz = 0;
} else {
nc = &nc_local;
}
/* init */
nc->curr_tok = RNCPRINT;
nc->number_value.d = 0.0;
nc->number_value.n = 0LL;
nc->errors = 0;
nc->oc = 0;
nc->calc_err = NULL;
nc->calc_i = 0;
nc->calc_len = 0;
nc->calc_buf = NULL;
nc->under_calc = true;
load_token (num, nc, str);
get_token (num, nc);
n = expr (num, nc, 0);
2017-02-01 22:20:20 +00:00
if (err) {
*err = nc->calc_err;
}
if (num) {
num->fvalue = n.d;
}
nc->under_calc = false;
return n.n;
}
#ifdef TEST
int main(int argc, char* argv[]) {
RNumCalcValue n;
RNumCalc nc;
while (!feof (stdin)) {
get_token (nc);
2017-02-01 22:20:20 +00:00
if (nc.curr_tok == RNCEND) {
break;
}
if (nc.curr_tok == RNCPRINT) {
continue;
}
n = expr (num, nc, 0);
2017-02-01 22:20:20 +00:00
if (n.d == ((double)(int)n.d)) {
printf ("%llx\n", n.n);
2017-02-01 22:20:20 +00:00
} else {
printf ("%lf\n", n.d);
}
}
return nc->errors;
}
#endif