diff options
-rw-r--r-- | jim.c | 1376 | ||||
-rw-r--r-- | jim.h | 2 | ||||
-rw-r--r-- | tests/expr-old.test | 849 | ||||
-rw-r--r-- | tests/expr.test | 44 |
4 files changed, 1768 insertions, 503 deletions
@@ -388,8 +388,9 @@ int Jim_StringToDouble(const char *str, double *doublePtr) char *endptr; *doublePtr = strtod(str, &endptr); - if (str[0] == '\0' || endptr[0] != '\0' || (str == endptr) ) + if (str[0] == '\0' || endptr[0] != '\0' || (str == endptr) ) { return JIM_ERR; + } return JIM_OK; } @@ -4755,10 +4756,9 @@ void UpdateStringOfInt(struct Jim_Obj *objPtr) int SetIntFromAny(Jim_Interp *interp, Jim_Obj *objPtr, int flags) { jim_wide wideValue; - const char *str; /* Get the string representation */ - str = Jim_GetString(objPtr, NULL); + const char *str = Jim_GetString(objPtr, NULL); /* Try to convert into a jim_wide */ if (Jim_StringToWide(str, &wideValue, 0) != JIM_OK) { if (flags & JIM_ERRMSG) { @@ -5973,7 +5973,7 @@ int Jim_SetDictKeysVector(Jim_Interp *interp, Jim_Obj *varNamePtr, Jim_Obj *varObjPtr, *objPtr, *dictObjPtr; int shared, i; - varObjPtr = objPtr = Jim_GetVariable(interp, varNamePtr, JIM_ERRMSG); + varObjPtr = objPtr = Jim_GetVariable(interp, varNamePtr, newObjPtr == NULL ? JIM_ERRMSG : JIM_NONE); if (objPtr == NULL) { if (newObjPtr == NULL) /* Cannot remove a key from non existing var */ return JIM_ERR; @@ -6190,111 +6190,694 @@ static int JimParseExprIrrational(struct JimParserCtx *pc); /* Exrp's Stack machine operators opcodes. */ /* Binary operators (numbers) */ -#define JIM_EXPROP_BINARY_NUM_FIRST 0 /* first */ -#define JIM_EXPROP_MUL 0 -#define JIM_EXPROP_DIV 1 -#define JIM_EXPROP_MOD 2 -#define JIM_EXPROP_SUB 3 -#define JIM_EXPROP_ADD 4 -#define JIM_EXPROP_LSHIFT 5 -#define JIM_EXPROP_RSHIFT 6 -#define JIM_EXPROP_ROTL 7 -#define JIM_EXPROP_ROTR 8 -#define JIM_EXPROP_LT 9 -#define JIM_EXPROP_GT 10 -#define JIM_EXPROP_LTE 11 -#define JIM_EXPROP_GTE 12 -#define JIM_EXPROP_NUMEQ 13 -#define JIM_EXPROP_NUMNE 14 -#define JIM_EXPROP_BITAND 15 -#define JIM_EXPROP_BITXOR 16 -#define JIM_EXPROP_BITOR 17 -#define JIM_EXPROP_LOGICAND 18 -#define JIM_EXPROP_LOGICOR 19 -#define JIM_EXPROP_LOGICAND_LEFT 20 -#define JIM_EXPROP_LOGICOR_LEFT 21 -#define JIM_EXPROP_POW 22 -#define JIM_EXPROP_BINARY_NUM_LAST 22 /* last */ +enum { + JIM_EXPROP_MUL, + JIM_EXPROP_DIV, + JIM_EXPROP_MOD, + JIM_EXPROP_SUB, + JIM_EXPROP_ADD, + JIM_EXPROP_LSHIFT, + JIM_EXPROP_RSHIFT, + JIM_EXPROP_ROTL, + JIM_EXPROP_ROTR, + JIM_EXPROP_LT, + JIM_EXPROP_GT, + JIM_EXPROP_LTE, + JIM_EXPROP_GTE, + JIM_EXPROP_NUMEQ, + JIM_EXPROP_NUMNE, + JIM_EXPROP_BITAND, + JIM_EXPROP_BITXOR, + JIM_EXPROP_BITOR, + + /* Note must keep these together */ + JIM_EXPROP_LOGICAND, + JIM_EXPROP_LOGICAND_LEFT, + JIM_EXPROP_LOGICAND_RIGHT, + + /* and these */ + JIM_EXPROP_LOGICOR, + JIM_EXPROP_LOGICOR_LEFT, + JIM_EXPROP_LOGICOR_RIGHT, + + /* and these */ + /* Ternary operators */ + JIM_EXPROP_TERNARY, + JIM_EXPROP_TERNARY_LEFT, + JIM_EXPROP_TERNARY_RIGHT, + + /* and these */ + JIM_EXPROP_COLON, + JIM_EXPROP_COLON_LEFT, + JIM_EXPROP_COLON_RIGHT, + + JIM_EXPROP_POW, /* Binary operators (strings) */ -#define JIM_EXPROP_STREQ 23 -#define JIM_EXPROP_STRNE 24 + JIM_EXPROP_STREQ, + JIM_EXPROP_STRNE, /* Unary operators (numbers) */ -#define JIM_EXPROP_NOT 25 -#define JIM_EXPROP_BITNOT 26 -#define JIM_EXPROP_UNARYMINUS 27 -#define JIM_EXPROP_UNARYPLUS 28 -#define JIM_EXPROP_LOGICAND_RIGHT 29 -#define JIM_EXPROP_LOGICOR_RIGHT 30 - -/* Ternary operators */ -#define JIM_EXPROP_TERNARY 31 -#define JIM_EXPROP_TERNARY_COLON 32 - -/* Operands */ -#define JIM_EXPROP_NUMBER 33 -#define JIM_EXPROP_COMMAND 34 -#define JIM_EXPROP_VARIABLE 35 -#define JIM_EXPROP_DICTSUGAR 36 -#define JIM_EXPROP_SUBST 37 -#define JIM_EXPROP_STRING 38 + JIM_EXPROP_NOT, + JIM_EXPROP_BITNOT, + JIM_EXPROP_UNARYMINUS, + JIM_EXPROP_UNARYPLUS, + + /* Operands */ + JIM_EXPROP_NUMBER, + JIM_EXPROP_COMMAND, + JIM_EXPROP_VARIABLE, + JIM_EXPROP_DICTSUGAR, + JIM_EXPROP_SUBST, + JIM_EXPROP_STRING, + + /* Functions */ + JIM_EXPROP_FUNC_INT, + JIM_EXPROP_FUNC_ABS, + JIM_EXPROP_FUNC_DOUBLE, + JIM_EXPROP_FUNC_ROUND, +}; + +struct expr_state { + Jim_Obj **stack; + int stacklen; + int opcode; + int skip; +}; + +typedef int jim_expr_function_t(Jim_Interp *interp, struct expr_state *e); /* Operators table */ typedef struct Jim_ExprOperator { const char *name; int precedence; int arity; - int opcode; + jim_expr_function_t *funcop; } Jim_ExprOperator; +#define GET_INT 1 +#define GET_DOUBLE 2 +#define GET_STRING 4 + +static Jim_Obj *expr_pop(Jim_Interp *interp, struct expr_state *e, int which) +{ + jim_wide wA; + double dA; + + if (e->stacklen > 0) { + Jim_Obj *obj = e->stack[e->stacklen - 1]; + + /* If it is already an integer or double, use it */ + if ((which & GET_INT) && obj->typePtr == &intObjType) { + e->stacklen--; + return obj; + } + /* Don't consider a double type with no string rep since it may + * have been explicitly converted. + */ + if ((which & GET_DOUBLE) && obj->typePtr == &doubleObjType && !obj->bytes) { + e->stacklen--; + return obj; + } + + /* Try to convert */ + if ((which & GET_INT) && Jim_GetWide(interp, obj, &wA) == JIM_OK) { + e->stacklen--; + return obj; + } + if ((which & GET_DOUBLE) && Jim_GetDouble(interp, obj, &dA) == JIM_OK) { + e->stacklen--; + return obj; + } + + /* Maybe a string is OK */ + if (which & GET_STRING) { + Jim_GetString(obj, NULL); + e->stacklen--; + return obj; + } + } + + /* Failure - leave it on the stack */ + return NULL; +} + +static void expr_push(struct expr_state *e, Jim_Obj *obj) +{ + Jim_IncrRefCount(obj); + e->stack[e->stacklen++] = obj; +} + + +static int JimExprOpNumUnary(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE); + + if (A) { + jim_wide wC; + double dC; + int intresult = 0; + + if (A->typePtr == &doubleObjType) { + double dA; + + Jim_GetDouble(interp, A, &dA); + switch (e->opcode) { + case JIM_EXPROP_FUNC_INT: wC = dA; intresult = 1; break; + case JIM_EXPROP_FUNC_ROUND: wC = dA < 0 ? (dA - 0.5) : (dA + 0.5); intresult = 1; break; + case JIM_EXPROP_FUNC_DOUBLE: dC = dA; break; + case JIM_EXPROP_FUNC_ABS: dC = dA >= 0 ? dA : -dA; break; + case JIM_EXPROP_UNARYMINUS: dC = -dA; break; + case JIM_EXPROP_UNARYPLUS: dC = dA; break; + case JIM_EXPROP_NOT: wC = !dA; intresult = 1; break; + default: abort(); + } + } + else { + /* Must be an integer */ + jim_wide wA; + + intresult = 1; + + Jim_GetWide(interp, A, &wA); + + switch (e->opcode) { + case JIM_EXPROP_FUNC_INT: wC = wA; break; + case JIM_EXPROP_FUNC_ROUND: wC = wA; break; + case JIM_EXPROP_FUNC_DOUBLE: dC = wA; intresult = 0; break; + case JIM_EXPROP_FUNC_ABS: wC = wA >= 0 ? wA : -wA; break; + case JIM_EXPROP_UNARYMINUS: wC = -wA; break; + case JIM_EXPROP_UNARYPLUS: wC = wA; break; + case JIM_EXPROP_NOT: wC = !wA; break; + default: abort(); + } + } + + if (intresult) { + expr_push(e, Jim_NewIntObj(interp, wC)); + } + else { + expr_push(e, Jim_NewDoubleObj(interp, dC)); + } + Jim_DecrRefCount(interp, A); + return JIM_OK; + } + + return JIM_ERR; +} + +static int JimExprOpIntUnary(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *A = expr_pop(interp, e, GET_INT); + + if (A) { + jim_wide wA; + jim_wide wC; + + Jim_GetWide(interp, A, &wA); + + switch (e->opcode) { + case JIM_EXPROP_BITNOT: wC = ~wA; break; + default: abort(); + } + expr_push(e, Jim_NewIntObj(interp, wC)); + Jim_DecrRefCount(interp, A); + return JIM_OK; + } + + return JIM_ERR; +} + +/* A binary operation on two ints */ +static int JimExprOpIntBin(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *B = expr_pop(interp, e, GET_INT); + Jim_Obj *A = expr_pop(interp, e, GET_INT); + + if (A && B) { + jim_wide wA, wB, wC; + int rc = JIM_OK; + + Jim_GetWide(interp, A, &wA); + Jim_GetWide(interp, B, &wB); + + switch (e->opcode) { + case JIM_EXPROP_LSHIFT: wC = wA<<wB; break; + case JIM_EXPROP_RSHIFT: wC = wA>>wB; break; + case JIM_EXPROP_BITAND: wC = wA&wB; break; + case JIM_EXPROP_BITXOR: wC = wA^wB; break; + case JIM_EXPROP_BITOR: wC = wA|wB; break; + case JIM_EXPROP_POW: wC = JimPowWide(wA,wB); break; + case JIM_EXPROP_MOD: + if (wB == 0) { + wC = 0; + Jim_SetResultString(interp, "Division by zero", -1); + rc = JIM_ERR; + } + else { + /* + * From Tcl 8.x + * + * This code is tricky: C doesn't guarantee much + * about the quotient or remainder, but Tcl does. + * The remainder always has the same sign as the + * divisor and a smaller absolute value. + */ + int negative = 0; + if (wB < 0) { + wB = -wB; + wA = -wA; + negative = 1; + } + wC = wA % wB; + if (wC < 0) { + wC += wB; + } + if (negative) { + wC = -wC; + } + } + break; + case JIM_EXPROP_ROTL: { + /* uint32_t would be better. But not everyone has inttypes.h?*/ + unsigned long uA = (unsigned long)wA; + const unsigned int S = sizeof(unsigned long) * 8; + wC = (unsigned long)((uA<<wB)|(uA>>(S-wB))); + break; + } + case JIM_EXPROP_ROTR: { + unsigned long uA = (unsigned long)wA; + const unsigned int S = sizeof(unsigned long) * 8; + wC = (unsigned long)((uA>>wB)|(uA<<(S-wB))); + break; + } + default: abort(); + } + expr_push(e, Jim_NewIntObj(interp, wC)); + + Jim_DecrRefCount(interp, A); + Jim_DecrRefCount(interp, B); + + return rc; + } + + if (A) { + Jim_DecrRefCount(interp, A); + } + if (B) { + Jim_DecrRefCount(interp, B); + } + + return JIM_ERR; +} + + +/* A binary operation on two ints or two doubles */ +static int JimExprOpNumBin(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *B = expr_pop(interp, e, GET_INT | GET_DOUBLE); + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE); + + if (A && B) { + int rc = JIM_OK; + jim_wide wC; + double dC; + int intresult = 0; + + /* If either is a double, the result is a double */ + if (A->typePtr == &doubleObjType || B->typePtr == &doubleObjType) { + double dA, dB; + + Jim_GetDouble(interp, A, &dA); + Jim_GetDouble(interp, B, &dB); + + switch (e->opcode) { + case JIM_EXPROP_ADD: dC = dA+dB; break; + case JIM_EXPROP_SUB: dC = dA-dB; break; + case JIM_EXPROP_MUL: dC = dA*dB; break; + case JIM_EXPROP_DIV: + if (dB == 0) { + dC = 0; + Jim_SetResultString(interp, "Division by zero", -1); + rc = JIM_ERR; + } + else { + dC = dA/dB; + } + break; + default: abort(); + } + } + else { + /* Must be both integers */ + jim_wide wA, wB; + + Jim_GetWide(interp, A, &wA); + Jim_GetWide(interp, B, &wB); + + intresult = 1; + + switch (e->opcode) { + case JIM_EXPROP_ADD: wC = wA+wB; break; + case JIM_EXPROP_SUB: wC = wA-wB; break; + case JIM_EXPROP_MUL: wC = wA*wB; break; + case JIM_EXPROP_DIV: + if (wB == 0) { + wC = 0; + Jim_SetResultString(interp, "Division by zero", -1); + rc = JIM_ERR; + } + else { + /* + * From Tcl 8.x + * + * This code is tricky: C doesn't guarantee much + * about the quotient or remainder, but Tcl does. + * The remainder always has the same sign as the + * divisor and a smaller absolute value. + */ + if (wB < 0) { + wB = -wB; + wA = -wA; + } + wC = wA / wB; + if (wA % wB < 0) { + wC--; + } + } + break; + default: abort(); + } + } + + if (intresult) { + expr_push(e, Jim_NewIntObj(interp, wC)); + } + else { + expr_push(e, Jim_NewDoubleObj(interp, dC)); + } + + Jim_DecrRefCount(interp, A); + Jim_DecrRefCount(interp, B); + + return rc; + } + + if (A) { + Jim_DecrRefCount(interp, A); + } + if (B) { + Jim_DecrRefCount(interp, B); + } + + return JIM_ERR; +} + +/* A binary operation on two ints, two doubles or two strings returning a boolean (int) */ +static int JimExprOpBin(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *B = expr_pop(interp, e, GET_INT | GET_DOUBLE | GET_STRING); + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE | GET_STRING); + + double dA, dB; + jim_wide wA, wB; + jim_wide wC; + + /* If either is a double, cooerce to doubles */ + if ((A->typePtr == &doubleObjType || B->typePtr == &doubleObjType) && + Jim_GetDouble(interp, A, &dA) == JIM_OK && Jim_GetDouble(interp, B, &dB) == JIM_OK) { + + switch (e->opcode) { + case JIM_EXPROP_LT: wC = dA<dB; break; + case JIM_EXPROP_GT: wC = dA>dB; break; + case JIM_EXPROP_LTE: wC = dA<=dB; break; + case JIM_EXPROP_GTE: wC = dA>=dB; break; + case JIM_EXPROP_NUMEQ: wC = dA==dB; break; + case JIM_EXPROP_NUMNE: wC = dA!=dB; break; + default: abort(); + } + } + /* Try ints */ + else if (JimGetWideNoErr(interp, A, &wA) == JIM_OK && JimGetWideNoErr(interp, B, &wB) == JIM_OK) { + + switch (e->opcode) { + case JIM_EXPROP_LT: wC = wA<wB; break; + case JIM_EXPROP_GT: wC = wA>wB; break; + case JIM_EXPROP_LTE: wC = wA<=wB; break; + case JIM_EXPROP_GTE: wC = wA>=wB; break; + case JIM_EXPROP_NUMEQ: wC = wA==wB; break; + case JIM_EXPROP_NUMNE: wC = wA!=wB; break; + default: abort(); + } + } + else { + /* Finally compare strings */ + int Alen, Blen; + + const char *sA = Jim_GetString(A, &Alen); + const char *sB = Jim_GetString(B, &Blen); + + switch(e->opcode) { + case JIM_EXPROP_LT: + wC = JimStringCompare(sA, Alen, sB, Blen, 0) < 0; break; + case JIM_EXPROP_GT: + wC = JimStringCompare(sA, Alen, sB, Blen, 0) > 0; break; + case JIM_EXPROP_LTE: + wC = JimStringCompare(sA, Alen, sB, Blen, 0) <= 0; break; + case JIM_EXPROP_GTE: + wC = JimStringCompare(sA, Alen, sB, Blen, 0) >= 0; break; + case JIM_EXPROP_NUMEQ: + wC = (Alen == Blen && memcmp(sA, sB, Alen) == 0); break; + case JIM_EXPROP_NUMNE: + wC = (Alen != Blen || memcmp(sA, sB, Alen) != 0); break; + default: abort(); + } + } + expr_push(e, Jim_NewIntObj(interp, wC)); + + Jim_DecrRefCount(interp, A); + Jim_DecrRefCount(interp, B); + + return JIM_OK; +} + +static int JimExprOpStrBin(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *B = expr_pop(interp, e, GET_STRING); + Jim_Obj *A = expr_pop(interp, e, GET_STRING); + + int Alen, Blen; + jim_wide wC; + + const char *sA = Jim_GetString(A, &Alen); + const char *sB = Jim_GetString(B, &Blen); + + switch(e->opcode) { + case JIM_EXPROP_STREQ: + wC = (Alen == Blen && memcmp(sA, sB, Alen) == 0); break; + case JIM_EXPROP_STRNE: + wC = (Alen != Blen || memcmp(sA, sB, Alen) != 0); break; + default: abort(); + } + expr_push(e, Jim_NewIntObj(interp, wC)); + + Jim_DecrRefCount(interp, A); + Jim_DecrRefCount(interp, B); + + return JIM_OK; +} + +static int expr_bool(Jim_Obj *obj) +{ + if (obj->typePtr == &doubleObjType) { + return obj->internalRep.doubleValue != 0; + } + else if (obj->typePtr == &intObjType) { + return obj->internalRep.wideValue != 0; + } + assert(0); +} + +static int JimExprOpAndLeft(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *skip = expr_pop(interp, e, GET_INT); + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE); + + if (skip && A) { + if (!expr_bool(A)) { + /* Failed, so skip RHS opcodes with a 0 result */ + e->skip = skip->internalRep.wideValue; + expr_push(e, Jim_NewIntObj(interp, 0)); + } + Jim_DecrRefCount(interp, skip); + Jim_DecrRefCount(interp, A); + return JIM_OK; + } + + if (skip) { + Jim_DecrRefCount(interp, skip); + } + + return JIM_ERR; +} + +static int JimExprOpOrLeft(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *skip = expr_pop(interp, e, GET_INT); + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE); + + if (skip && A) { + if (expr_bool(A)) { + /* Succeeded, so skip RHS opcodes with a 1 result */ + e->skip = skip->internalRep.wideValue; + expr_push(e, Jim_NewIntObj(interp, 1)); + } + Jim_DecrRefCount(interp, skip); + Jim_DecrRefCount(interp, A); + return JIM_OK; + } + + if (skip) { + Jim_DecrRefCount(interp, skip); + } + + assert(A == NULL); + + return JIM_ERR; +} + +static int JimExprOpAndOrRight(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE); + + if (A) { + expr_push(e, Jim_NewIntObj(interp, expr_bool(A))); + Jim_DecrRefCount(interp, A); + return JIM_OK; + } + + return JIM_ERR; +} + +static int JimExprOpTernaryLeft(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *skip = expr_pop(interp, e, GET_INT); + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE); + + if (skip && A) { + /* Repush A */ + expr_push(e, A); + if (!expr_bool(A)) { + /* Failed, so skip RHS opcodes */ + e->skip = skip->internalRep.wideValue; + /* Push a dummy value */ + expr_push(e, Jim_NewIntObj(interp, 0)); + } + Jim_DecrRefCount(interp, skip); + Jim_DecrRefCount(interp, A); + return JIM_OK; + } + + if (skip) { + Jim_DecrRefCount(interp, skip); + } + + return JIM_ERR; +} + +static int JimExprOpColonLeft(Jim_Interp *interp, struct expr_state *e) +{ + Jim_Obj *skip = expr_pop(interp, e, GET_INT); + Jim_Obj *B = expr_pop(interp, e, GET_INT | GET_DOUBLE | GET_STRING); + Jim_Obj *A = expr_pop(interp, e, GET_INT | GET_DOUBLE); + + if (skip && A && B) { + if (expr_bool(A)) { + /* Success, so skip RHS opcodes */ + e->skip = skip->internalRep.wideValue; + /* Repush B as the answer */ + expr_push(e, B); + } + Jim_DecrRefCount(interp, skip); + Jim_DecrRefCount(interp, A); + Jim_DecrRefCount(interp, B); + return JIM_OK; + } + + if (skip) { + Jim_DecrRefCount(interp, skip); + } + if (A) { + Jim_DecrRefCount(interp, A); + } + if (B) { + Jim_DecrRefCount(interp, B); + } + + return JIM_ERR; +} + +static int JimExprOpNull(Jim_Interp *interp, struct expr_state *e) +{ + return JIM_OK; +} + /* name - precedence - arity - opcode */ static const struct Jim_ExprOperator Jim_ExprOperators[] = { - {"!", 300, 1, JIM_EXPROP_NOT}, - {"~", 300, 1, JIM_EXPROP_BITNOT}, - {"unarymin", 300, 1, JIM_EXPROP_UNARYMINUS}, -/* {"unaryplus", 300, 1, JIM_EXPROP_UNARYPLUS}, */ + [JIM_EXPROP_FUNC_INT] = {"int", 400, 1, JimExprOpNumUnary }, + [JIM_EXPROP_FUNC_DOUBLE] = {"double", 400, 1, JimExprOpNumUnary }, + [JIM_EXPROP_FUNC_ABS] = {"abs", 400, 1, JimExprOpNumUnary }, + [JIM_EXPROP_FUNC_ROUND] = {"round", 400, 1, JimExprOpNumUnary }, - {"**", 250, 2, JIM_EXPROP_POW}, + [JIM_EXPROP_NOT] = {"!", 300, 1, JimExprOpNumUnary }, + [JIM_EXPROP_BITNOT] = {"~", 300, 1, JimExprOpIntUnary }, + [JIM_EXPROP_UNARYMINUS] = {"unarymin", 300, 1, JimExprOpNumUnary }, + [JIM_EXPROP_UNARYPLUS] = {"unaryplus", 300, 1, JimExprOpNumUnary }, - {"*", 200, 2, JIM_EXPROP_MUL}, - {"/", 200, 2, JIM_EXPROP_DIV}, - {"%", 200, 2, JIM_EXPROP_MOD}, + [JIM_EXPROP_POW] = {"**", 250, 2, JimExprOpIntBin }, - {"-", 100, 2, JIM_EXPROP_SUB}, - {"+", 100, 2, JIM_EXPROP_ADD}, + [JIM_EXPROP_MUL] = {"*", 200, 2, JimExprOpNumBin }, + [JIM_EXPROP_DIV] = {"/", 200, 2, JimExprOpNumBin }, + [JIM_EXPROP_MOD] = {"%", 200, 2, JimExprOpIntBin }, - {"<<<", 90, 3, JIM_EXPROP_ROTL}, - {">>>", 90, 3, JIM_EXPROP_ROTR}, - {"<<", 90, 2, JIM_EXPROP_LSHIFT}, - {">>", 90, 2, JIM_EXPROP_RSHIFT}, + [JIM_EXPROP_SUB] = {"-", 100, 2, JimExprOpNumBin }, + [JIM_EXPROP_ADD] = {"+", 100, 2, JimExprOpNumBin }, - {"<", 80, 2, JIM_EXPROP_LT}, - {">", 80, 2, JIM_EXPROP_GT}, - {"<=", 80, 2, JIM_EXPROP_LTE}, - {">=", 80, 2, JIM_EXPROP_GTE}, + [JIM_EXPROP_ROTL] = {"<<<", 90, 2, JimExprOpIntBin }, + [JIM_EXPROP_ROTR] = {">>>", 90, 2, JimExprOpIntBin }, + [JIM_EXPROP_LSHIFT] = {"<<", 90, 2, JimExprOpIntBin }, + [JIM_EXPROP_RSHIFT] = {">>", 90, 2, JimExprOpIntBin }, - {"==", 70, 2, JIM_EXPROP_NUMEQ}, - {"!=", 70, 2, JIM_EXPROP_NUMNE}, + [JIM_EXPROP_LT] = {"<", 80, 2, JimExprOpBin }, + [JIM_EXPROP_GT] = {">", 80, 2, JimExprOpBin }, + [JIM_EXPROP_LTE] = {"<=", 80, 2, JimExprOpBin }, + [JIM_EXPROP_GTE] = {">=", 80, 2, JimExprOpBin }, - {"eq", 60, 2, JIM_EXPROP_STREQ}, - {"ne", 60, 2, JIM_EXPROP_STRNE}, + [JIM_EXPROP_NUMEQ] = {"==", 70, 2, JimExprOpBin }, + [JIM_EXPROP_NUMNE] = {"!=", 70, 2, JimExprOpBin }, - {"&", 50, 2, JIM_EXPROP_BITAND}, - {"^", 49, 2, JIM_EXPROP_BITXOR}, - {"|", 48, 2, JIM_EXPROP_BITOR}, + [JIM_EXPROP_STREQ] = {"eq", 60, 2, JimExprOpStrBin }, + [JIM_EXPROP_STRNE] = {"ne", 60, 2, JimExprOpStrBin }, - {"&&", 10, 2, JIM_EXPROP_LOGICAND}, - {"||", 10, 2, JIM_EXPROP_LOGICOR}, + [JIM_EXPROP_BITAND] = {"&", 50, 2, JimExprOpIntBin }, + [JIM_EXPROP_BITXOR] = {"^", 49, 2, JimExprOpIntBin }, + [JIM_EXPROP_BITOR] = {"|", 48, 2, JimExprOpIntBin }, - {"?", 5, 3, JIM_EXPROP_TERNARY}, - {":", 5, 1, JIM_EXPROP_TERNARY_COLON}, + [JIM_EXPROP_LOGICAND] = {"&&", 10, 2, }, + [JIM_EXPROP_LOGICOR] = {"||", 9, 2, }, + + [JIM_EXPROP_TERNARY] = {"?", 5, 2, }, + [JIM_EXPROP_COLON] = {":", 5, 2, }, /* private operators */ - {NULL, 10, 2, JIM_EXPROP_LOGICAND_LEFT}, - {NULL, 10, 1, JIM_EXPROP_LOGICAND_RIGHT}, - {NULL, 10, 2, JIM_EXPROP_LOGICOR_LEFT}, - {NULL, 10, 1, JIM_EXPROP_LOGICOR_RIGHT}, + [JIM_EXPROP_TERNARY_LEFT] = {NULL, 5, 2, JimExprOpTernaryLeft }, + [JIM_EXPROP_TERNARY_RIGHT] = {NULL, 5, 0, JimExprOpNull }, + [JIM_EXPROP_COLON_LEFT] = {NULL, 5, 3, JimExprOpColonLeft }, + [JIM_EXPROP_COLON_RIGHT] = {NULL, 5, 0, JimExprOpNull }, + [JIM_EXPROP_LOGICAND_LEFT] = {NULL, 10, 2, JimExprOpAndLeft }, + [JIM_EXPROP_LOGICAND_RIGHT] = {NULL, 10, 1, JimExprOpAndOrRight }, + [JIM_EXPROP_LOGICOR_LEFT] = {NULL, 9, 2, JimExprOpOrLeft }, + [JIM_EXPROP_LOGICOR_RIGHT] = {NULL, 9, 1, JimExprOpAndOrRight }, }; #define JIM_EXPR_OPERATORS_NUM \ @@ -6393,7 +6976,7 @@ int JimParseExprNumber(struct JimParserCtx *pc) if (*pc->p == '.') allowdot = 0; pc->p++; pc->len--; - if (!allowdot && *pc->p == 'e' && *(pc->p+1) == '-') { + if (!allowhex && (*pc->p == 'e' || *pc->p == 'E') && (pc->p[1] == '-' || pc->p[1] == '+' || isdigit(pc->p[1]))) { pc->p += 2; pc->len -= 2; } } @@ -6431,7 +7014,9 @@ int JimParseExprOperator(struct JimParserCtx *pc) int oplen; opname = Jim_ExprOperators[i].name; - if (opname == NULL) continue; + if (opname == NULL) { + continue; + } oplen = strlen(opname); if (strncmp(opname, pc->p, oplen) == 0 && oplen > bestLen) { @@ -6439,7 +7024,25 @@ int JimParseExprOperator(struct JimParserCtx *pc) bestLen = oplen; } } - if (bestIdx == -1) return JIM_ERR; + if (bestIdx == -1) { + return JIM_ERR; + } + + /* Could validate paretheses around function arguments, or just not bother. Thus 'int 1.5' is OK */ +#if 0 + if (Jim_ExprOperators[bestIdx].opcode >= JIM_EXPROP_FUNC_FIRST && Jim_ExprOperators[bestIdx].opcode <= JIM_EXPROP_FUNC_LAST) { + /* We expect an open parethesis after a function */ + const char *p = pc->p + bestLen; + int len = pc->len - bestLen; + while (len && isspace(*p)) { + len--; + p++; + } + if (*p != '(') { + return JIM_ERR; + } + } +#endif pc->tstart = pc->p; pc->tend = pc->p + bestLen - 1; pc->p += bestLen; pc->len -= bestLen; @@ -6448,7 +7051,7 @@ int JimParseExprOperator(struct JimParserCtx *pc) return JIM_OK; } -const struct Jim_ExprOperator *JimExprOperatorInfo(const char *opname) +static const struct Jim_ExprOperator *JimExprOperatorInfo(const char *opname) { int i; for (i = 0; i < (signed)JIM_EXPR_OPERATORS_NUM; i++) @@ -6458,15 +7061,17 @@ const struct Jim_ExprOperator *JimExprOperatorInfo(const char *opname) return NULL; } -const struct Jim_ExprOperator *JimExprOperatorInfoByOpcode(int opcode) +static const struct Jim_ExprOperator *JimExprOperatorInfoByOpcode(int opcode) { - int i; - for (i = 0; i < (signed)JIM_EXPR_OPERATORS_NUM; i++) - if (Jim_ExprOperators[i].opcode == opcode) - return &Jim_ExprOperators[i]; - return NULL; + return &Jim_ExprOperators[opcode]; +} + +static int JimExprOperatorOpcode(const struct Jim_ExprOperator *op) +{ + return op - &Jim_ExprOperators[0]; } + /* ----------------------------------------------------------------------------- * Expression Object * ---------------------------------------------------------------------------*/ @@ -6520,7 +7125,23 @@ static void ExprObjAddInstr(Jim_Interp *interp, ExprByteCode *expr, expr->opcode = Jim_Realloc(expr->opcode, sizeof(int)*(expr->len+1)); expr->obj = Jim_Realloc(expr->obj, sizeof(Jim_Obj*)*(expr->len+1)); expr->opcode[expr->len] = opcode; - expr->obj[expr->len] = Jim_NewStringObjNoAlloc(interp, str, len); + if (opcode == JIM_EXPROP_NUMBER) { + /* Special handling for numbers. Immediately convert */ + jim_wide w; + if (Jim_StringToWide(str, &w, 0) == JIM_OK) { + expr->obj[expr->len] = Jim_NewIntObj(interp, w); + } + else { + double d; + + Jim_StringToDouble(str, &d); + expr->obj[expr->len] = Jim_NewDoubleObj(interp, d); + } + Jim_Free(str); + } + else { + expr->obj[expr->len] = Jim_NewStringObjNoAlloc(interp, str, len); + } Jim_IncrRefCount(expr->obj[expr->len]); expr->len++; } @@ -6530,6 +7151,8 @@ static int ExprCheckCorrectness(ExprByteCode *expr) { int i; int stacklen = 0; + int ternary = 0; + const struct Jim_ExprOperator *op; /* Try to check if there are stack underflows, * and make sure at the end of the program there is @@ -6544,49 +7167,26 @@ static int ExprCheckCorrectness(ExprByteCode *expr) case JIM_EXPROP_COMMAND: stacklen++; break; - case JIM_EXPROP_NOT: - case JIM_EXPROP_BITNOT: - case JIM_EXPROP_UNARYMINUS: - case JIM_EXPROP_UNARYPLUS: - /* Unary operations */ - if (stacklen < 1) return JIM_ERR; - break; - case JIM_EXPROP_ADD: - case JIM_EXPROP_SUB: - case JIM_EXPROP_MUL: - case JIM_EXPROP_DIV: - case JIM_EXPROP_MOD: - case JIM_EXPROP_LT: - case JIM_EXPROP_GT: - case JIM_EXPROP_LTE: - case JIM_EXPROP_GTE: - case JIM_EXPROP_ROTL: - case JIM_EXPROP_ROTR: - case JIM_EXPROP_LSHIFT: - case JIM_EXPROP_RSHIFT: - case JIM_EXPROP_NUMEQ: - case JIM_EXPROP_NUMNE: - case JIM_EXPROP_STREQ: - case JIM_EXPROP_STRNE: - case JIM_EXPROP_BITAND: - case JIM_EXPROP_BITXOR: - case JIM_EXPROP_BITOR: - case JIM_EXPROP_LOGICAND: - case JIM_EXPROP_LOGICOR: - case JIM_EXPROP_POW: - case JIM_EXPROP_TERNARY_COLON: - case JIM_EXPROP_TERNARY: - /* binary operations */ - if (stacklen < 2) return JIM_ERR; - stacklen--; - break; default: - Jim_Panic(NULL,"Default opcode reached ExprCheckCorrectness"); + op = JimExprOperatorInfoByOpcode(expr->opcode[i]); + stacklen -= op->arity; + if (stacklen < 0) { + return JIM_ERR; + } + stacklen++; + if (JimExprOperatorOpcode(op) == JIM_EXPROP_TERNARY) { + ternary++; + } + else if (JimExprOperatorOpcode(op) == JIM_EXPROP_COLON) { + ternary--; + } break; } } - if (stacklen != 1) return JIM_ERR; + if (stacklen != 1 || ternary != 0) { + return JIM_ERR; + } return JIM_OK; } @@ -6637,23 +7237,31 @@ static void ExprShareLiterals(Jim_Interp *interp, ExprByteCode *expr, */ static void ExprMakeLazy(Jim_Interp *interp, ExprByteCode *expr) { + int i; + while (1) { - int index = -1, leftindex, arity, i, offset; + int index = -1, leftindex, arity, offset; const Jim_ExprOperator *op; /* Search for || or && */ - for (i = 0; i < expr->len; i++) { - if (expr->opcode[i] == JIM_EXPROP_LOGICAND || - expr->opcode[i] == JIM_EXPROP_LOGICOR) { + for (i = expr->len - 1; i >= 0; i--) { + if (expr->opcode[i] == JIM_EXPROP_LOGICAND + || expr->opcode[i] == JIM_EXPROP_LOGICOR + || expr->opcode[i] == JIM_EXPROP_TERNARY + || expr->opcode[i] == JIM_EXPROP_COLON + ) { index = i; break; } } - if (index == -1) return; + if (index == -1) { + break; + } + /* Search for the end of the first operator */ leftindex = index-1; arity = 1; - while(arity) { + while (arity) { switch(expr->opcode[leftindex]) { case JIM_EXPROP_NUMBER: case JIM_EXPROP_COMMAND: @@ -6676,6 +7284,7 @@ static void ExprMakeLazy(Jim_Interp *interp, ExprByteCode *expr) leftindex--; } leftindex++; + expr->opcode = Jim_Realloc(expr->opcode, sizeof(int)*(expr->len+2)); expr->obj = Jim_Realloc(expr->obj, sizeof(Jim_Obj*)*(expr->len+2)); memmove(&expr->opcode[leftindex+2], &expr->opcode[leftindex], @@ -6686,25 +7295,62 @@ static void ExprMakeLazy(Jim_Interp *interp, ExprByteCode *expr) index += 2; offset = (index-leftindex)-1; Jim_DecrRefCount(interp, expr->obj[index]); - if (expr->opcode[index] == JIM_EXPROP_LOGICAND) { - expr->opcode[leftindex+1] = JIM_EXPROP_LOGICAND_LEFT; - expr->opcode[index] = JIM_EXPROP_LOGICAND_RIGHT; - expr->obj[leftindex+1] = Jim_NewStringObj(interp, "&L", -1); - expr->obj[index] = Jim_NewStringObj(interp, "&R", -1); - } else { - expr->opcode[leftindex+1] = JIM_EXPROP_LOGICOR_LEFT; - expr->opcode[index] = JIM_EXPROP_LOGICOR_RIGHT; - expr->obj[leftindex+1] = Jim_NewStringObj(interp, "|L", -1); - expr->obj[index] = Jim_NewStringObj(interp, "|R", -1); - } + + /* Now we rely on the fact the the left and right version have opcodes + * 1 and 2 after the main opcode respectively + */ + expr->opcode[leftindex+1] = expr->opcode[index] + 1; + expr->opcode[index] = expr->opcode[index] + 2; + expr->obj[leftindex+1] = Jim_NewStringObj(interp, "lazyL", -1); + expr->obj[index] = Jim_NewStringObj(interp, "lazyR", -1); + expr->opcode[leftindex] = JIM_EXPROP_NUMBER; expr->obj[leftindex] = Jim_NewIntObj(interp, offset); Jim_IncrRefCount(expr->obj[index]); Jim_IncrRefCount(expr->obj[leftindex]); Jim_IncrRefCount(expr->obj[leftindex+1]); + + /* Do we need to adjust the skip count for any &L, |L, ?L or :L in the left operand? */ + for (i = leftindex - 1; i > 0; i--) { + if (expr->opcode[i] == JIM_EXPROP_LOGICOR_LEFT + || expr->opcode[i] == JIM_EXPROP_LOGICAND_LEFT + || expr->opcode[i] == JIM_EXPROP_TERNARY_LEFT + || expr->opcode[i] == JIM_EXPROP_COLON_LEFT + ) { + long skip; + + Jim_GetLong(interp, expr->obj[i - 1], &skip); + if (skip + i - 1 >= leftindex) { + Jim_DecrRefCount(interp, expr->obj[i - 1]); + expr->obj[i - 1] = Jim_NewIntObj(interp, skip + 2); + Jim_IncrRefCount(expr->obj[i - 1]); + } + } + } } } +static int expr_reduce_stack(Jim_Interp *interp, ExprByteCode *expr, Jim_Stack *stack, const char *token) +{ + while (Jim_StackLen(stack)) { + const struct Jim_ExprOperator *op; + + char *opstr = Jim_StackPop(stack); + if (token && strcmp(opstr, token) == 0) { + Jim_Free(opstr); + return 1; + } + op = JimExprOperatorInfo(opstr); + if (op == NULL) { + Jim_Free(opstr); + return -1; + } + ExprObjAddInstr(interp, expr, JimExprOperatorOpcode(op), opstr, -1); + } + + return 0; +} + /* This method takes the string representation of an expression * and generates a program for the Expr's stack-based VM. */ int SetExprFromAny(Jim_Interp *interp, struct Jim_Obj *objPtr) @@ -6764,10 +7410,19 @@ int SetExprFromAny(Jim_Interp *interp, struct Jim_Obj *objPtr) break; case JIM_TT_EXPR_OPERATOR: /* Convert - to unary minus if necessary */ - if (*token == '-' && (prevtt == JIM_TT_NONE || prevtt == JIM_TT_EXPR_OPERATOR)) { - Jim_Free(token); - token = Jim_StrDup("unarymin"); + if (prevtt == JIM_TT_NONE || prevtt == JIM_TT_EXPR_OPERATOR) { + if (*token == '-') { + Jim_Free(token); + token = Jim_StrDup("unarymin"); + parser.tt = JIM_TT_NONE; + } + else if (*token == '+') { + Jim_Free(token); + token = Jim_StrDup("unaryplus"); + parser.tt = JIM_TT_NONE; + } } + op = JimExprOperatorInfo(token); while(1) { @@ -6778,10 +7433,14 @@ int SetExprFromAny(Jim_Interp *interp, struct Jim_Obj *objPtr) } else { stackTopOp = NULL; } + + /* XXX: Should handle right-to-left associativity of ?: operator */ + /* XXX: ? without :, : without ? */ + if (Jim_StackLen(&stack) && op->arity != 1 && stackTopOp && stackTopOp->precedence >= op->precedence) { - ExprObjAddInstr(interp, expr, stackTopOp->opcode, + ExprObjAddInstr(interp, expr, JimExprOperatorOpcode(stackTopOp), Jim_StackPeek(&stack), -1); Jim_StackPop(&stack); } else { @@ -6793,46 +7452,34 @@ int SetExprFromAny(Jim_Interp *interp, struct Jim_Obj *objPtr) case JIM_TT_SUBEXPR_START: Jim_StackPush(&stack, Jim_StrDup("(")); Jim_Free(token); + parser.tt = JIM_TT_NONE; break; case JIM_TT_SUBEXPR_END: - { - int found = 0; - while(Jim_StackLen(&stack)) { - char *opstr = Jim_StackPop(&stack); - if (!strcmp(opstr, "(")) { - Jim_Free(opstr); - found = 1; - break; - } - op = JimExprOperatorInfo(opstr); - ExprObjAddInstr(interp, expr, op->opcode, opstr, -1); - } - if (!found) { - Jim_SetResultString(interp, - "Unexpected close parenthesis", -1); - goto err; - } - } Jim_Free(token); + if (expr_reduce_stack(interp, expr, &stack, "(") != 1) { + Jim_SetResultString(interp, + "Unexpected close parenthesis", -1); + goto err; + } break; default: Jim_Panic(interp,"Default reached in SetExprFromAny()"); break; } } - while (Jim_StackLen(&stack)) { - char *opstr = Jim_StackPop(&stack); - op = JimExprOperatorInfo(opstr); - if (op == NULL) { - Jim_Free(opstr); + + if (Jim_StackLen(&stack)) { + if (expr_reduce_stack(interp, expr, &stack, NULL) != 0) { Jim_SetResultString(interp, "Missing close parenthesis", -1); goto err; } - ExprObjAddInstr(interp, expr, op->opcode, opstr, -1); } + /* Check program correctness. */ if (ExprCheckCorrectness(expr) != JIM_OK) { - Jim_SetResultString(interp, "Invalid expression", -1); + Jim_SetResultString(interp, "Invalid expression: ", -1); + Jim_AppendStrings(interp, Jim_GetResult(interp), + Jim_GetString(objPtr, NULL), NULL); goto err; } @@ -6900,8 +7547,9 @@ int Jim_EvalExpression(Jim_Interp *interp, Jim_Obj *exprObjPtr, Jim_Obj **exprResultPtrPtr) { ExprByteCode *expr; - Jim_Obj **stack, *staticStack[JIM_EE_STATICSTACK_LEN]; - int stacklen = 0, i, error = 0, errRetCode = JIM_ERR; + Jim_Obj *staticStack[JIM_EE_STATICSTACK_LEN]; + int i, error = 0, errRetCode = JIM_ERR; + struct expr_state e; Jim_IncrRefCount(exprObjPtr); expr = Jim_GetExpression(interp, exprObjPtr); @@ -6920,351 +7568,80 @@ int Jim_EvalExpression(Jim_Interp *interp, Jim_Obj *exprObjPtr, * a program of length N can't require a stack longer than * N. */ if (expr->len > JIM_EE_STATICSTACK_LEN) - stack = Jim_Alloc(sizeof(Jim_Obj*)*expr->len); + e.stack = Jim_Alloc(sizeof(Jim_Obj*)*expr->len); else - stack = staticStack; + e.stack = staticStack; + + e.stacklen = 0; /* Execute every instruction */ - for (i = 0; i < expr->len; i++) { - Jim_Obj *A, *B, *C, *objPtr; - jim_wide wA, wB, wC; - double dA, dB, dC; - const char *sA, *sB; - int Alen, Blen, retcode; + for (i = 0; i < expr->len && !error; i++) { + Jim_Obj *objPtr; + int retcode; int opcode = expr->opcode[i]; - /* Is the result of the double expression an int (wC)? */ - int intresult; + const struct Jim_ExprOperator *opinfo = JimExprOperatorInfoByOpcode(opcode); + + e.skip = 0; + e.opcode = opcode; + + if (e.stacklen < opinfo->arity) { + Jim_Panic(interp, "Reached end of expr stack prematurely"); + } + if (opinfo->funcop) { + errRetCode = opinfo->funcop(interp, &e); + if (errRetCode != JIM_OK) { + error = 1; + } + else { + i += e.skip; + } + continue; + } - if (opcode == JIM_EXPROP_NUMBER || opcode == JIM_EXPROP_STRING) { - stack[stacklen++] = expr->obj[i]; - Jim_IncrRefCount(expr->obj[i]); + if (opcode == JIM_EXPROP_NUMBER) { + expr_push(&e, expr->obj[i]); + } + else if (opcode == JIM_EXPROP_STRING) { + expr_push(&e, expr->obj[i]); } else if (opcode == JIM_EXPROP_VARIABLE) { objPtr = Jim_GetVariable(interp, expr->obj[i], JIM_ERRMSG); if (objPtr == NULL) { error = 1; - goto err; } - stack[stacklen++] = objPtr; - Jim_IncrRefCount(objPtr); + else { + expr_push(&e, objPtr); + } } else if (opcode == JIM_EXPROP_SUBST) { if ((retcode = Jim_SubstObj(interp, expr->obj[i], &objPtr, JIM_NONE)) != JIM_OK) { error = 1; errRetCode = retcode; - goto err; } - stack[stacklen++] = objPtr; - Jim_IncrRefCount(objPtr); + else { + expr_push(&e, objPtr); + } } else if (opcode == JIM_EXPROP_DICTSUGAR) { objPtr = Jim_ExpandDictSugar(interp, expr->obj[i]); if (objPtr == NULL) { error = 1; - goto err; } - stack[stacklen++] = objPtr; - Jim_IncrRefCount(objPtr); + else { + expr_push(&e, objPtr); + } } else if (opcode == JIM_EXPROP_COMMAND) { if ((retcode = Jim_EvalObj(interp, expr->obj[i])) != JIM_OK) { error = 1; errRetCode = retcode; - goto err; - } - stack[stacklen++] = interp->result; - Jim_IncrRefCount(interp->result); - } else if (opcode >= JIM_EXPROP_BINARY_NUM_FIRST && - opcode <= JIM_EXPROP_BINARY_NUM_LAST) - { - /* Note that there isn't to increment the - * refcount of objects. the references are moved - * from stack to A and B. */ - B = stack[--stacklen]; - A = stack[--stacklen]; - - /* --- Integer --- */ - if ((A->typePtr == &doubleObjType && !A->bytes) || - (B->typePtr == &doubleObjType && !B->bytes) || - JimGetWideNoErr(interp, A, &wA) != JIM_OK || - JimGetWideNoErr(interp, B, &wB) != JIM_OK) { - goto trydouble; - } - Jim_DecrRefCount(interp, A); - Jim_DecrRefCount(interp, B); - switch(expr->opcode[i]) { - case JIM_EXPROP_ADD: wC = wA+wB; break; - case JIM_EXPROP_SUB: wC = wA-wB; break; - case JIM_EXPROP_MUL: wC = wA*wB; break; - case JIM_EXPROP_LT: wC = wA<wB; break; - case JIM_EXPROP_GT: wC = wA>wB; break; - case JIM_EXPROP_LTE: wC = wA<=wB; break; - case JIM_EXPROP_GTE: wC = wA>=wB; break; - case JIM_EXPROP_LSHIFT: wC = wA<<wB; break; - case JIM_EXPROP_RSHIFT: wC = wA>>wB; break; - case JIM_EXPROP_NUMEQ: wC = wA==wB; break; - case JIM_EXPROP_NUMNE: wC = wA!=wB; break; - case JIM_EXPROP_BITAND: wC = wA&wB; break; - case JIM_EXPROP_BITXOR: wC = wA^wB; break; - case JIM_EXPROP_BITOR: wC = wA|wB; break; - case JIM_EXPROP_POW: wC = JimPowWide(wA,wB); break; - case JIM_EXPROP_LOGICAND_LEFT: - if (wA == 0) { - i += (int)wB; - wC = 0; - } else { - continue; - } - break; - case JIM_EXPROP_LOGICOR_LEFT: - if (wA != 0) { - i += (int)wB; - wC = 1; - } else { - continue; - } - break; - case JIM_EXPROP_DIV: - if (wB == 0) goto divbyzero; - wC = wA/wB; - break; - case JIM_EXPROP_MOD: - if (wB == 0) goto divbyzero; - wC = wA%wB; - break; - case JIM_EXPROP_ROTL: { - /* uint32_t would be better. But not everyone has inttypes.h?*/ - unsigned long uA = (unsigned long)wA; - const unsigned int S = sizeof(unsigned long) * 8; - wC = (unsigned long)((uA<<wB)|(uA>>(S-wB))); - break; - } - case JIM_EXPROP_ROTR: { - unsigned long uA = (unsigned long)wA; - const unsigned int S = sizeof(unsigned long) * 8; - wC = (unsigned long)((uA>>wB)|(uA<<(S-wB))); - break; - } - - default: - wC = 0; /* avoid gcc warning */ - break; - } - stack[stacklen] = Jim_NewIntObj(interp, wC); - Jim_IncrRefCount(stack[stacklen]); - stacklen++; - continue; -trydouble: - /* --- Double --- */ - if (Jim_GetDouble(interp, A, &dA) != JIM_OK || - Jim_GetDouble(interp, B, &dB) != JIM_OK) { - - /* Hmmm! For compatibility, maybe convert != and == into ne and eq */ - if (expr->opcode[i] == JIM_EXPROP_NUMNE) { - opcode = JIM_EXPROP_STRNE; - goto retry_as_string; - } - else if (expr->opcode[i] == JIM_EXPROP_NUMEQ) { - opcode = JIM_EXPROP_STREQ; - goto retry_as_string; - } - goto retry_as_string; - } - intresult = 0; - Jim_DecrRefCount(interp, A); - Jim_DecrRefCount(interp, B); - switch(expr->opcode[i]) { - case JIM_EXPROP_ROTL: - case JIM_EXPROP_ROTR: - case JIM_EXPROP_LSHIFT: - case JIM_EXPROP_RSHIFT: - case JIM_EXPROP_BITAND: - case JIM_EXPROP_BITXOR: - case JIM_EXPROP_BITOR: - case JIM_EXPROP_MOD: - case JIM_EXPROP_POW: - Jim_SetResultString(interp, - "Got floating-point value where integer was expected", -1); - error = 1; - goto err; - break; - case JIM_EXPROP_ADD: dC = dA+dB; break; - case JIM_EXPROP_SUB: dC = dA-dB; break; - case JIM_EXPROP_MUL: dC = dA*dB; break; - case JIM_EXPROP_LT: wC = dA<dB; intresult = 1; break; - case JIM_EXPROP_GT: wC = dA>dB; intresult = 1; break; - case JIM_EXPROP_LTE: wC = dA<=dB; intresult = 1; break; - case JIM_EXPROP_GTE: wC = dA>=dB; intresult = 1; break; - case JIM_EXPROP_NUMEQ: wC = dA==dB; intresult = 1; break; - case JIM_EXPROP_NUMNE: wC = dA!=dB; intresult = 1; break; - case JIM_EXPROP_LOGICAND_LEFT: - if (dA == 0) { - i += (int)dB; - dC = 0; - } else { - continue; - } - break; - case JIM_EXPROP_LOGICOR_LEFT: - if (dA != 0) { - i += (int)dB; - dC = 1; - } else { - continue; - } - break; - case JIM_EXPROP_DIV: - if (dB == 0) goto divbyzero; - dC = dA/dB; - break; - default: - dC = 0; /* avoid gcc warning */ - break; - } - if (intresult) { - stack[stacklen] = Jim_NewIntObj(interp, wC); } else { - stack[stacklen] = Jim_NewDoubleObj(interp, dC); + expr_push(&e, Jim_GetResult(interp)); } - Jim_IncrRefCount(stack[stacklen]); - stacklen++; - } else if (opcode == JIM_EXPROP_STREQ || opcode == JIM_EXPROP_STRNE) { - B = stack[--stacklen]; - A = stack[--stacklen]; -retry_as_string: - sA = Jim_GetString(A, &Alen); - sB = Jim_GetString(B, &Blen); - switch(opcode) { - case JIM_EXPROP_LT: - wC = JimStringCompare(sA, Alen, sB, Blen, 0) < 0; break; - case JIM_EXPROP_GT: - wC = JimStringCompare(sA, Alen, sB, Blen, 0) > 0; break; - case JIM_EXPROP_LTE: - wC = JimStringCompare(sA, Alen, sB, Blen, 0) <= 0; break; - case JIM_EXPROP_GTE: - wC = JimStringCompare(sA, Alen, sB, Blen, 0) >= 0; break; - case JIM_EXPROP_STREQ: - if (Alen == Blen && memcmp(sA, sB, Alen) ==0) - wC = 1; - else - wC = 0; - break; - case JIM_EXPROP_STRNE: - if (Alen != Blen || memcmp(sA, sB, Alen) != 0) - wC = 1; - else - wC = 0; - break; - default: - /* Not a valid string comparison */ - Jim_DecrRefCount(interp, A); - Jim_DecrRefCount(interp, B); - error = 1; - goto err; - } - intresult = 0; - Jim_DecrRefCount(interp, A); - Jim_DecrRefCount(interp, B); - stack[stacklen] = Jim_NewIntObj(interp, wC); - Jim_IncrRefCount(stack[stacklen]); - stacklen++; - } else if (opcode == JIM_EXPROP_NOT || - opcode == JIM_EXPROP_BITNOT || - opcode == JIM_EXPROP_UNARYMINUS || - opcode == JIM_EXPROP_LOGICAND_RIGHT || - opcode == JIM_EXPROP_LOGICOR_RIGHT) { - /* Note that there isn't to increment the - * refcount of objects. the references are moved - * from stack to A and B. */ - A = stack[--stacklen]; - - /* --- Integer --- */ - if ((A->typePtr == &doubleObjType && !A->bytes) || - JimGetWideNoErr(interp, A, &wA) != JIM_OK) { - goto trydouble_unary; - } - Jim_DecrRefCount(interp, A); - switch(expr->opcode[i]) { - case JIM_EXPROP_NOT: wC = !wA; break; - case JIM_EXPROP_BITNOT: wC = ~wA; break; - case JIM_EXPROP_UNARYMINUS: wC = -wA; break; - case JIM_EXPROP_LOGICAND_RIGHT: - case JIM_EXPROP_LOGICOR_RIGHT: wC = (wA != 0); break; - default: - wC = 0; /* avoid gcc warning */ - break; - } - stack[stacklen] = Jim_NewIntObj(interp, wC); - Jim_IncrRefCount(stack[stacklen]); - stacklen++; - continue; -trydouble_unary: - /* --- Double --- */ - if (Jim_GetDouble(interp, A, &dA) != JIM_OK) { - Jim_DecrRefCount(interp, A); - error = 1; - goto err; - } - Jim_DecrRefCount(interp, A); - switch(expr->opcode[i]) { - case JIM_EXPROP_NOT: dC = !dA; break; - case JIM_EXPROP_UNARYMINUS: dC = -dA; break; - case JIM_EXPROP_LOGICAND_RIGHT: - case JIM_EXPROP_LOGICOR_RIGHT: dC = (dA != 0); break; - case JIM_EXPROP_BITNOT: - Jim_SetResultString(interp, - "Got floating-point value where integer was expected", -1); - error = 1; - goto err; - break; - default: - dC = 0; /* avoid gcc warning */ - break; - } - stack[stacklen] = Jim_NewDoubleObj(interp, dC); - Jim_IncrRefCount(stack[stacklen]); - stacklen++; - } else if (opcode == JIM_EXPROP_TERNARY) { - /* ? operator */ - C = stack[--stacklen]; /* false option */ - B = stack[--stacklen]; /* true option */ - A = stack[--stacklen]; /* condition */ - - if (JimGetWideNoErr(interp, A, &wA) != JIM_OK) { - if (Jim_GetDouble(interp, A, &dA) == JIM_OK) { - wA = (dA != 0); - } - else { - stacklen += 3; - error = 1; - Jim_SetResultString(interp, - "Wrong type for ternary operator", -1); - goto err; - } - } - if (wA) { - stack[stacklen] = B; - } - else { - stack[stacklen] = C; - } - - Jim_IncrRefCount(stack[stacklen]); - stacklen++; - - /* skip the TERNARY_COLON operator */ - - Jim_DecrRefCount(interp, A); - Jim_DecrRefCount(interp, B); - Jim_DecrRefCount(interp, C); - continue; - } else if (opcode == JIM_EXPROP_TERNARY_COLON) { - /* Nothing to do! */ } else { Jim_Panic(interp,"Unknown opcode in Jim_EvalExpression (%d)", opcode); } } -err: /* There is no need to decerement the inUse field because * this reference is transfered back into the exprObjPtr. */ Jim_FreeIntRep(interp, exprObjPtr); @@ -7272,20 +7649,16 @@ err: Jim_SetIntRepPtr(exprObjPtr, expr); Jim_DecrRefCount(interp, exprObjPtr); if (!error) { - *exprResultPtrPtr = stack[0]; - Jim_IncrRefCount(stack[0]); + *exprResultPtrPtr = e.stack[0]; + Jim_IncrRefCount(e.stack[0]); errRetCode = JIM_OK; } - for (i = 0; i < stacklen; i++) { - Jim_DecrRefCount(interp, stack[i]); + for (i = 0; i < e.stacklen; i++) { + Jim_DecrRefCount(interp, e.stack[i]); } - if (stack != staticStack) - Jim_Free(stack); + if (e.stack != staticStack) + Jim_Free(e.stack); return errRetCode; -divbyzero: - error = 1; - Jim_SetResultString(interp, "Division by zero", -1); - goto err; } int Jim_GetBoolFromExpr(Jim_Interp *interp, Jim_Obj *exprObjPtr, int *boolPtr) @@ -7298,6 +7671,7 @@ int Jim_GetBoolFromExpr(Jim_Interp *interp, Jim_Obj *exprObjPtr, int *boolPtr) retcode = Jim_EvalExpression(interp, exprObjPtr, &exprResultPtr); if (retcode != JIM_OK) return retcode; + if (JimGetWideNoErr(interp, exprResultPtr, &wideValue) != JIM_OK) { if (Jim_GetDouble(interp, exprResultPtr, &doubleValue) != JIM_OK) { @@ -135,7 +135,7 @@ extern "C" { /* Jim version numbering: every version of jim is marked with a * successive integer number. This is version 0. The first * stable version will be 1, then 2, 3, and so on. */ -#define JIM_VERSION 61 +#define JIM_VERSION 62 #define JIM_OK 0 #define JIM_ERR 1 diff --git a/tests/expr-old.test b/tests/expr-old.test new file mode 100644 index 0000000..41c6139 --- /dev/null +++ b/tests/expr-old.test @@ -0,0 +1,849 @@ +# Commands covered: expr +# +# This file contains the original set of tests for Tcl's expr command. +# Since the expr command is now compiled, a new set of tests covering +# the new implementation are in the files "parseExpr.test and +# "compExpr.test". Sourcing this file into Tcl runs the tests and generates +# output for errors. No output means no errors were found. +# +# Copyright (c) 1991-1994 The Regents of the University of California. +# Copyright (c) 1994-1997 Sun Microsystems, Inc. +# Copyright (c) 1998-1999 by Scriptics Corporation. +# +# See the file "license.terms" for information on usage and redistribution +# of this file, and for a DISCLAIMER OF ALL WARRANTIES. +# +# RCS: @(#) $Id: expr-old.test,v 1.8.2.1 2002/04/18 13:10:27 msofer Exp $ + +source testing.tcl + +# First, test all of the integer operators individually. + +test expr-old-1.1 {integer operators} {expr -4} -4 +test expr-old-1.2 {integer operators} {expr -(1+4)} -5 +test expr-old-1.3 {integer operators} {expr ~3} -4 +test expr-old-1.4 {integer operators} {expr !2} 0 +test expr-old-1.5 {integer operators} {expr !0} 1 +test expr-old-1.6 {integer operators} {expr 4*6} 24 +test expr-old-1.7 {integer operators} {expr 36/12} 3 +test expr-old-1.8 {integer operators} {expr 27/4} 6 +test expr-old-1.9 {integer operators} {expr 27%4} 3 +test expr-old-1.10 {integer operators} {expr 2+2} 4 +test expr-old-1.11 {integer operators} {expr 2-6} -4 +test expr-old-1.12 {integer operators} {expr 1<<3} 8 +test expr-old-1.13 {integer operators} {expr 0xff>>2} 63 +test expr-old-1.14 {integer operators} {expr -1>>2} -1 +test expr-old-1.15 {integer operators} {expr 3>2} 1 +test expr-old-1.16 {integer operators} {expr 2>2} 0 +test expr-old-1.17 {integer operators} {expr 1>2} 0 +test expr-old-1.18 {integer operators} {expr 3<2} 0 +test expr-old-1.19 {integer operators} {expr 2<2} 0 +test expr-old-1.20 {integer operators} {expr 1<2} 1 +test expr-old-1.21 {integer operators} {expr 3>=2} 1 +test expr-old-1.22 {integer operators} {expr 2>=2} 1 +test expr-old-1.23 {integer operators} {expr 1>=2} 0 +test expr-old-1.24 {integer operators} {expr 3<=2} 0 +test expr-old-1.25 {integer operators} {expr 2<=2} 1 +test expr-old-1.26 {integer operators} {expr 1<=2} 1 +test expr-old-1.27 {integer operators} {expr 3==2} 0 +test expr-old-1.28 {integer operators} {expr 2==2} 1 +test expr-old-1.29 {integer operators} {expr 3!=2} 1 +test expr-old-1.30 {integer operators} {expr 2!=2} 0 +test expr-old-1.31 {integer operators} {expr 7&0x13} 3 +test expr-old-1.32 {integer operators} {expr 7^0x13} 20 +test expr-old-1.33 {integer operators} {expr 7|0x13} 23 +test expr-old-1.34 {integer operators} {expr 0&&1} 0 +test expr-old-1.35 {integer operators} {expr 0&&0} 0 +test expr-old-1.36 {integer operators} {expr 1&&3} 1 +test expr-old-1.37 {integer operators} {expr 0||1} 1 +test expr-old-1.38 {integer operators} {expr 3||0} 1 +test expr-old-1.39 {integer operators} {expr 0||0} 0 +test expr-old-1.40 {integer operators} {expr 3>2?44:66} 44 +test expr-old-1.41 {integer operators} {expr 2>3?44:66} 66 +test expr-old-1.42 {integer operators} {expr 36/5} 7 +test expr-old-1.43 {integer operators} {expr 36%5} 1 +test expr-old-1.44 {integer operators} {expr -36/5} -8 +test expr-old-1.45 {integer operators} {expr -36%5} 4 +test expr-old-1.46 {integer operators} {expr 36/-5} -8 +test expr-old-1.47 {integer operators} {expr 36%-5} -4 +test expr-old-1.48 {integer operators} {expr -36/-5} 7 +test expr-old-1.49 {integer operators} {expr -36%-5} -1 +test expr-old-1.50 {integer operators} {expr +36} 36 +test expr-old-1.51 {integer operators} {expr +--++36} 36 +test expr-old-1.52 {integer operators} {expr +36%+5} 1 +test expr-old-1.53 {integer operators} { + catch {unset x} + set x 1 + list [expr {1 && $x}] [expr {$x && 1}] \ + [expr {0 || $x}] [expr {$x || 0}] +} {1 1 1 1} + +# Check the floating-point operators individually, along with +# automatic conversion to integers where needed. + +test expr-old-2.1 {floating-point operators} {expr -4.2} -4.2 +test expr-old-2.2 {floating-point operators} {expr -(1.1+4.2)} -5.3 +test expr-old-2.3 {floating-point operators} {expr +5.7} 5.7 +test expr-old-2.4 {floating-point operators} {expr +--+-62.0} -62.0 +test expr-old-2.5 {floating-point operators} {expr !2.1} 0 +test expr-old-2.6 {floating-point operators} {expr !0.0} 1 +test expr-old-2.7 {floating-point operators} {expr 4.2*6.3} 26.46 +test expr-old-2.8 {floating-point operators} {expr 36.0/12.0} 3.0 +test expr-old-2.9 {floating-point operators} {expr 27/4.0} 6.75 +test expr-old-2.10 {floating-point operators} {expr 2.3+2.1} 4.4 +test expr-old-2.11 {floating-point operators} {expr 2.3-6.5} -4.2 +test expr-old-2.12 {floating-point operators} {expr 3.1>2.1} 1 +test expr-old-2.13 {floating-point operators} {expr {2.1 > 2.1}} 0 +test expr-old-2.14 {floating-point operators} {expr 1.23>2.34e+1} 0 +test expr-old-2.15 {floating-point operators} {expr 3.45<2.34} 0 +test expr-old-2.16 {floating-point operators} {expr 0.002e3<--200e-2} 0 +test expr-old-2.17 {floating-point operators} {expr 1.1<2.1} 1 +test expr-old-2.18 {floating-point operators} {expr 3.1>=2.2} 1 +test expr-old-2.19 {floating-point operators} {expr 2.345>=2.345} 1 +test expr-old-2.20 {floating-point operators} {expr 1.1>=2.2} 0 +test expr-old-2.21 {floating-point operators} {expr 3.0<=2.0} 0 +test expr-old-2.22 {floating-point operators} {expr 2.2<=2.2} 1 +test expr-old-2.23 {floating-point operators} {expr 2.2<=2.2001} 1 +test expr-old-2.24 {floating-point operators} {expr 3.2==2.2} 0 +test expr-old-2.25 {floating-point operators} {expr 2.2==2.2} 1 +test expr-old-2.26 {floating-point operators} {expr 3.2!=2.2} 1 +test expr-old-2.27 {floating-point operators} {expr 2.2!=2.2} 0 +test expr-old-2.28 {floating-point operators} {expr 0.0&&0.0} 0 +test expr-old-2.29 {floating-point operators} {expr 0.0&&1.3} 0 +test expr-old-2.30 {floating-point operators} {expr 1.3&&0.0} 0 +test expr-old-2.31 {floating-point operators} {expr 1.3&&3.3} 1 +test expr-old-2.32 {floating-point operators} {expr 0.0||0.0} 0 +test expr-old-2.33 {floating-point operators} {expr 0.0||1.3} 1 +test expr-old-2.34 {floating-point operators} {expr 1.3||0.0} 1 +test expr-old-2.35 {floating-point operators} {expr 3.3||0.0} 1 +test expr-old-2.36 {floating-point operators} {expr 3.3>2.3?44.3:66.3} 44.3 +test expr-old-2.37 {floating-point operators} {expr 2.3>3.3?44.3:66.3} 66.3 +test expr-old-2.38 {floating-point operators} { + list [catch {expr 028.1 + 09.2} msg] $msg +} {0 37.3} + +# Operators that aren't legal on floating-point numbers + +test expr-old-3.1 {illegal floating-point operations} { + list [catch {expr ~4.0} msg] +} {1} +test expr-old-3.2 {illegal floating-point operations} { + list [catch {expr 27%4.0} msg] +} {1} +test expr-old-3.3 {illegal floating-point operations} { + list [catch {expr 27.0%4} msg] +} {1} +test expr-old-3.4 {illegal floating-point operations} { + list [catch {expr 1.0<<3} msg] +} {1} +test expr-old-3.5 {illegal floating-point operations} { + list [catch {expr 3<<1.0} msg] +} {1} +test expr-old-3.6 {illegal floating-point operations} { + list [catch {expr 24.0>>3} msg] +} {1} +test expr-old-3.7 {illegal floating-point operations} { + list [catch {expr 24>>3.0} msg] +} {1} +test expr-old-3.8 {illegal floating-point operations} { + list [catch {expr 24&3.0} msg] +} {1} +test expr-old-3.9 {illegal floating-point operations} { + list [catch {expr 24.0|3} msg] +} {1} +test expr-old-3.10 {illegal floating-point operations} { + list [catch {expr 24.0^3} msg] +} {1} + +# Check the string operators individually. + +test expr-old-4.1 {string operators} {expr {"abc" > "def"}} 0 +test expr-old-4.2 {string operators} {expr {"def" > "def"}} 0 +test expr-old-4.3 {string operators} {expr {"g" > "def"}} 1 +test expr-old-4.4 {string operators} {expr {"abc" < "abd"}} 1 +test expr-old-4.5 {string operators} {expr {"abd" < "abd"}} 0 +test expr-old-4.6 {string operators} {expr {"abe" < "abd"}} 0 +test expr-old-4.7 {string operators} {expr {"abc" >= "def"}} 0 +test expr-old-4.8 {string operators} {expr {"def" >= "def"}} 1 +test expr-old-4.9 {string operators} {expr {"g" >= "def"}} 1 +test expr-old-4.10 {string operators} {expr {"abc" <= "abd"}} 1 +test expr-old-4.11 {string operators} {expr {"abd" <= "abd"}} 1 +test expr-old-4.12 {string operators} {expr {"abe" <= "abd"}} 0 +test expr-old-4.13 {string operators} {expr {"abc" == "abd"}} 0 +test expr-old-4.14 {string operators} {expr {"abd" == "abd"}} 1 +test expr-old-4.15 {string operators} {expr {"abc" != "abd"}} 1 +test expr-old-4.16 {string operators} {expr {"abd" != "abd"}} 0 +test expr-old-4.17 {string operators} {expr {"0y" < "0x12"}} 0 +test expr-old-4.18 {string operators} {expr {"." < " "}} 0 + +# The following tests are non-portable because on some systems "+" +# and "-" can be parsed as numbers. + +test expr-old-4.19 {string operators} {expr {"0" == "+"}} 0 +test expr-old-4.20 {string operators} {expr {"0" == "-"}} 0 +test expr-old-4.21 {string operators} {expr {1?"foo":"bar"}} foo +test expr-old-4.22 {string operators} {expr {0?"foo":"bar"}} bar + +# Operators that aren't legal on string operands. + +test expr-old-5.1 {illegal string operations} { + list [catch {expr {-"a"}} msg] +} {1} +test expr-old-5.2 {illegal string operations} { + list [catch {expr {+"a"}} msg] +} {1} +test expr-old-5.3 {illegal string operations} { + list [catch {expr {~"a"}} msg] +} {1} +test expr-old-5.4 {illegal string operations} { + list [catch {expr {!"a"}} msg] +} {1} +test expr-old-5.5 {illegal string operations} { + list [catch {expr {"a"*"b"}} msg] +} {1} +test expr-old-5.6 {illegal string operations} { + list [catch {expr {"a"/"b"}} msg] +} {1} +test expr-old-5.7 {illegal string operations} { + list [catch {expr {"a"%"b"}} msg] +} {1} +test expr-old-5.8 {illegal string operations} { + list [catch {expr {"a"+"b"}} msg] +} {1} +test expr-old-5.9 {illegal string operations} { + list [catch {expr {"a"-"b"}} msg] +} {1} +test expr-old-5.10 {illegal string operations} { + list [catch {expr {"a"<<"b"}} msg] +} {1} +test expr-old-5.11 {illegal string operations} { + list [catch {expr {"a">>"b"}} msg] +} {1} +test expr-old-5.12 {illegal string operations} { + list [catch {expr {"a"&"b"}} msg] +} {1} +test expr-old-5.13 {illegal string operations} { + list [catch {expr {"a"^"b"}} msg] +} {1} +test expr-old-5.14 {illegal string operations} { + list [catch {expr {"a"|"b"}} msg] +} {1} +test expr-old-5.15 {illegal string operations} { + list [catch {expr {"a"&&"b"}} msg] +} {1} +test expr-old-5.16 {illegal string operations} { + list [catch {expr {"a"||"b"}} msg] +} {1} +test expr-old-5.17 {illegal string operations} { + list [catch {expr {"a"?4:2}} msg] +} {1} + +# Check precedence pairwise. + +test expr-old-6.1 {precedence checks} {expr -~3} 4 +test expr-old-6.2 {precedence checks} {expr -!3} 0 +test expr-old-6.3 {precedence checks} {expr -~0} 1 + +test expr-old-7.1 {precedence checks} {expr 2*4/6} 1 +test expr-old-7.2 {precedence checks} {expr 24/6*3} 12 +test expr-old-7.3 {precedence checks} {expr 24/6/2} 2 + +test expr-old-8.1 {precedence checks} {expr -2+4} 2 +test expr-old-8.2 {precedence checks} {expr -2-4} -6 +test expr-old-8.3 {precedence checks} {expr +2-4} -2 + +test expr-old-9.1 {precedence checks} {expr 2*3+4} 10 +test expr-old-9.2 {precedence checks} {expr 8/2+4} 8 +test expr-old-9.3 {precedence checks} {expr 8%3+4} 6 +test expr-old-9.4 {precedence checks} {expr 2*3-1} 5 +test expr-old-9.5 {precedence checks} {expr 8/2-1} 3 +test expr-old-9.6 {precedence checks} {expr 8%3-1} 1 + +test expr-old-10.1 {precedence checks} {expr 6-3-2} 1 + +test expr-old-11.1 {precedence checks} {expr 7+1>>2} 2 +test expr-old-11.2 {precedence checks} {expr 7+1<<2} 32 +test expr-old-11.3 {precedence checks} {expr 7>>3-2} 3 +test expr-old-11.4 {precedence checks} {expr 7<<3-2} 14 + +test expr-old-12.1 {precedence checks} {expr 6>>1>4} 0 +test expr-old-12.2 {precedence checks} {expr 6>>1<2} 0 +test expr-old-12.3 {precedence checks} {expr 6>>1>=3} 1 +test expr-old-12.4 {precedence checks} {expr 6>>1<=2} 0 +test expr-old-12.5 {precedence checks} {expr 6<<1>5} 1 +test expr-old-12.6 {precedence checks} {expr 6<<1<5} 0 +test expr-old-12.7 {precedence checks} {expr 5<=6<<1} 1 +test expr-old-12.8 {precedence checks} {expr 5>=6<<1} 0 + +test expr-old-13.1 {precedence checks} {expr 2<3<4} 1 +test expr-old-13.2 {precedence checks} {expr 0<4>2} 0 +test expr-old-13.3 {precedence checks} {expr 4>2<1} 0 +test expr-old-13.4 {precedence checks} {expr 4>3>2} 0 +test expr-old-13.5 {precedence checks} {expr 4>3>=2} 0 +test expr-old-13.6 {precedence checks} {expr 4>=3>2} 0 +test expr-old-13.7 {precedence checks} {expr 4>=3>=2} 0 +test expr-old-13.8 {precedence checks} {expr 0<=4>=2} 0 +test expr-old-13.9 {precedence checks} {expr 4>=2<=0} 0 +test expr-old-13.10 {precedence checks} {expr 2<=3<=4} 1 + +test expr-old-14.1 {precedence checks} {expr 1==4>3} 1 +test expr-old-14.2 {precedence checks} {expr 0!=4>3} 1 +test expr-old-14.3 {precedence checks} {expr 1==3<4} 1 +test expr-old-14.4 {precedence checks} {expr 0!=3<4} 1 +test expr-old-14.5 {precedence checks} {expr 1==4>=3} 1 +test expr-old-14.6 {precedence checks} {expr 0!=4>=3} 1 +test expr-old-14.7 {precedence checks} {expr 1==3<=4} 1 +test expr-old-14.8 {precedence checks} {expr 0!=3<=4} 1 + +test expr-old-15.1 {precedence checks} {expr 1==3==3} 0 +test expr-old-15.2 {precedence checks} {expr 3==3!=2} 1 +test expr-old-15.3 {precedence checks} {expr 2!=3==3} 0 +test expr-old-15.4 {precedence checks} {expr 2!=1!=1} 0 + +test expr-old-16.1 {precedence checks} {expr 2&3==2} 0 +test expr-old-16.2 {precedence checks} {expr 1&3!=3} 0 + +test expr-old-17.1 {precedence checks} {expr 7&3^0x10} 19 +test expr-old-17.2 {precedence checks} {expr 7^0x10&3} 7 + +test expr-old-18.1 {precedence checks} {expr 7^0x10|3} 23 +test expr-old-18.2 {precedence checks} {expr 7|0x10^3} 23 + +test expr-old-19.1 {precedence checks} {expr 7|3&&1} 1 +test expr-old-19.2 {precedence checks} {expr 1&&3|7} 1 +test expr-old-19.3 {precedence checks} {expr 0&&1||1} 1 +test expr-old-19.4 {precedence checks} {expr 1||1&&0} 1 + +test expr-old-20.1 {precedence checks} {expr 1||0?3:4} 3 +test expr-old-20.2 {precedence checks} {expr 1?0:4||1} 0 +# Note: tinytcl gets the following horribly wrong +# at least jim only gets 20.3 wrong +test expr-old-20.3 {precedence checks} {expr 1?2:0?3:4} 2 +test expr-old-20.4 {precedence checks} {expr 0?2:0?3:4} 4 +test expr-old-20.5 {precedence checks} {expr 1?2?3:4:0} 3 +test expr-old-20.6 {precedence checks} {expr 0?2?3:4:0} 0 + +# Parentheses. + +test expr-old-21.1 {parenthesization} {expr (2+4)*6} 36 +test expr-old-21.2 {parenthesization} {expr (1?0:4)||1} 1 +test expr-old-21.3 {parenthesization} {expr +(3-4)} -1 + +# Embedded commands and variable names. + +set a 16 +test expr-old-22.1 {embedded variables} {expr {2*$a}} 32 +test expr-old-22.2 {embedded variables} { + set x -5 + set y 10 + expr {$x + $y} +} {5} +test expr-old-22.3 {embedded variables} { + set x " -5" + set y " +10" + expr {$x + $y} +} {5} +test expr-old-22.4 {embedded commands and variables} {expr {[set a] - 14}} 2 +test expr-old-22.5 {embedded commands and variables} { + list [catch {expr {12 - [bad_command_name]}} msg] $msg +} {1 {invalid command name "bad_command_name"}} + +# Double-quotes and things inside them. + +test expr-old-23.1 {double quotes} {expr {"abc"}} abc +test expr-old-23.2 {double quotes} { + set a 189 + expr {"$a.bc"} +} 189.bc +test expr-old-23.3 {double quotes} { + set b2 xyx + expr {"$b2$b2$b2.[set b2].[set b2]"} +} xyxxyxxyx.xyx.xyx +test expr-old-23.4 {double quotes} {expr {"11\}\}22"}} 11}}22 +test expr-old-23.5 {double quotes} {expr {"\*bc"}} {*bc} +test expr-old-23.6 {double quotes} { + catch {unset bogus__} + list [catch {expr {"$bogus__"}} msg] $msg +} {1 {can't read "bogus__": no such variable}} +test expr-old-23.7 {double quotes} { + list [catch {expr {"a[error Testing]bc"}} msg] $msg +} {1 Testing} +test expr-old-23.8 {double quotes} { + list [catch {expr {"12398712938788234-1298379" != ""}} msg] $msg +} {0 1} + +# Numbers in various bases. + +test expr-old-24.1 {numbers in different bases} {expr 0x20} 32 +test expr-old-24.2 {numbers in different bases} {expr 015} 13 + +# Conversions between various data types. + +test expr-old-25.1 {type conversions} {expr 2+2.5} 4.5 +test expr-old-25.2 {type conversions} {expr 2.5+2} 4.5 +test expr-old-25.3 {type conversions} {expr 2-2.5} -0.5 +test expr-old-25.4 {type conversions} {expr 2/2.5} 0.8 +test expr-old-25.5 {type conversions} {expr 2>2.5} 0 +test expr-old-25.6 {type conversions} {expr 2.5>2} 1 +test expr-old-25.7 {type conversions} {expr 2<2.5} 1 +test expr-old-25.8 {type conversions} {expr 2>=2.5} 0 +test expr-old-25.9 {type conversions} {expr 2<=2.5} 1 +test expr-old-25.10 {type conversions} {expr 2==2.5} 0 +test expr-old-25.11 {type conversions} {expr 2!=2.5} 1 +test expr-old-25.12 {type conversions} {expr 2>"ab"} 0 +test expr-old-25.13 {type conversions} {expr {2>" "}} 1 +test expr-old-25.14 {type conversions} {expr {"24.1a" > 24.1}} 1 +test expr-old-25.15 {type conversions} {expr {24.1 > "24.1a"}} 0 +test expr-old-25.16 {type conversions} {expr 2+2.5} 4.5 +test expr-old-25.17 {type conversions} {expr 2+2.5} 4.5 +test expr-old-25.18 {type conversions} {expr 2.0e2} 200.0 +test expr-old-25.19 {type conversions} {expr 2.0e30} 2e+30 +test expr-old-25.20 {type conversions} {expr 10.0} 10.0 + +# Various error conditions. + +test expr-old-26.1 {error conditions} { + list [catch {expr 2+"a"} msg] +} {1} +test expr-old-26.2 {error conditions} { + list [catch {expr 2+4*} msg] +} {1} +test expr-old-26.3 {error conditions} { + list [catch {expr 2+4*(} msg] +} {1} +catch {unset _non_existent_} +test expr-old-26.4 {error conditions} { + list [catch {expr 2+$_non_existent_} msg] +} {1} +set a xx +test expr-old-26.5 {error conditions} { + list [catch {expr {2+$a}} msg] +} {1} +test expr-old-26.6 {error conditions} { + list [catch {expr {2+[set a]}} msg] +} {1} +test expr-old-26.7 {error conditions} { + list [catch {expr {2+(4}} msg] +} {1} +test expr-old-26.8 {error conditions} { + list [catch {expr 2/0} msg] +} {1} +test expr-old-26.9 {error conditions} { + list [catch {expr 2%0} msg] +} {1} +test expr-old-26.10 {error conditions} { + list [catch {expr 2.0/0.0} msg] +} {1} +test expr-old-26.11 {error conditions} { + list [catch {expr 2#} msg] +} {1} +test expr-old-26.12 {error conditions} { + list [catch {expr a.b} msg] +} {1} +test expr-old-26.13 {error conditions} { + list [catch {expr {"a"/"b"}} msg] +} {1} +test expr-old-26.14 {error conditions} { + list [catch {expr 2:3} msg] +} {1} +test expr-old-26.15 {error conditions} { + list [catch {expr a@b} msg] +} {1} +test expr-old-26.16 {error conditions} { + list [catch {expr a[b} msg] +} {1} +test expr-old-26.17 {error conditions} { + list [catch {expr a`b} msg] +} {1} +test expr-old-26.18 {error conditions} { + list [catch {expr \"a\"\{b} msg] +} {1} +test expr-old-26.19 {error conditions} { + list [catch {expr a} msg] +} {1} +test expr-old-26.20 {error conditions} { + list [catch expr msg] +} {1} + +# Cancelled evaluation. + +test expr-old-27.1 {cancelled evaluation} { + set a 1 + expr {0&&[set a 2]} + set a +} 1 +test expr-old-27.2 {cancelled evaluation} { + set a 1 + expr {1||[set a 2]} + set a +} 1 +test expr-old-27.3 {cancelled evaluation} { + set a 1 + expr {0?[set a 2]:1} + set a +} 1 +test expr-old-27.4 {cancelled evaluation} { + set a 1 + expr {1?2:[set a 2]} + set a +} 1 +catch {unset x} +test expr-old-27.5 {cancelled evaluation} { + list [catch {expr {[info exists x] && $x}} msg] $msg +} {0 0} +test expr-old-27.6 {cancelled evaluation} { + list [catch {expr {0 && [concat $x]}} msg] $msg +} {0 0} +test expr-old-27.7 {cancelled evaluation} { + set one 1 + list [catch {expr {1 || 1/$one}} msg] $msg +} {0 1} +test expr-old-27.8 {cancelled evaluation} { + list [catch {expr {1 || -"string"}} msg] $msg +} {0 1} +test expr-old-27.9 {cancelled evaluation} { + list [catch {expr {1 || ("string" * ("x" && "y"))}} msg] $msg +} {0 1} +test expr-old-27.10 {cancelled evaluation} { + set x -1.3 + list [catch {expr {($x > 0) ? round($x) : 0}} msg] $msg +} {0 0} +test expr-old-27.11 {cancelled evaluation} { + list [catch {expr {0 && foo}} msg] +} {1} +test expr-old-27.12 {cancelled evaluation} { + list [catch {expr {0 ? 1 : foo}} msg] +} {1} + +# Operands enclosed in braces + +#test expr-old-29.1 {braces} {expr {{abc}}} abc +#test expr-old-29.2 {braces} {expr {{00010}}} 8 +#test expr-old-29.3 {braces} {expr {{3.1200000}}} 3.12 +#test expr-old-29.4 {braces} {expr {{a{b}{1 {2 3}}c}}} "a{b}{1 {2 3}}c" +#test expr-old-29.5 {braces} { +# list [catch {expr "\{abc"} msg] $msg +#} {1 {missing close-brace}} + +# Very long values + +test expr-old-30.1 {long values} { + set a "0000 1111 2222 3333 4444" + set a "$a | $a | $a | $a | $a" + set a "$a || $a || $a || $a || $a" + expr {$a} +} {0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 || 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 || 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 || 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 || 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444 | 0000 1111 2222 3333 4444} +test expr-old-30.2 {long values} { + set a "000000000000000000000000000000" + set a "$a$a$a$a$a$a$a$a$a$a$a$a$a$a$a$a${a}5" + expr $a +} 5 + +# Expressions spanning multiple arguments + +test expr-old-31.1 {multiple arguments to expr command} { + expr 4 + ( 6 *12) -3 +} 73 +test expr-old-31.2 {multiple arguments to expr command} { + list [catch {expr 2 + (3 + 4} msg] +} {1} +test expr-old-31.3 {multiple arguments to expr command} { + list [catch {expr 2 + 3 +} msg] +} {1} +test expr-old-31.4 {multiple arguments to expr command} { + list [catch {expr 2 + 3 )} msg] +} {1} + +# Math functions + +if {0} { +test expr-old-32.1 {math functions in expressions} { + format %.6g [expr acos(0.5)] +} {1.0472} +test expr-old-32.2 {math functions in expressions} { + format %.6g [expr asin(0.5)] +} {0.523599} +test expr-old-32.3 {math functions in expressions} { + format %.6g [expr atan(1.0)] +} {0.785398} +test expr-old-32.4 {math functions in expressions} { + format %.6g [expr atan2(2.0, 2.0)] +} {0.785398} +test expr-old-32.5 {math functions in expressions} { + format %.6g [expr ceil(1.999)] +} {2} +test expr-old-32.6 {math functions in expressions} { + format %.6g [expr cos(.1)] +} {0.995004} +test expr-old-32.7 {math functions in expressions} { + format %.6g [expr cosh(.1)] +} {1.005} +test expr-old-32.8 {math functions in expressions} { + format %.6g [expr exp(1.0)] +} {2.71828} +test expr-old-32.9 {math functions in expressions} { + format %.6g [expr floor(2.000)] +} {2} +test expr-old-32.10 {math functions in expressions} { + format %.6g [expr floor(2.001)] +} {2} +test expr-old-32.11 {math functions in expressions} { + format %.6g [expr fmod(7.3, 3.2)] +} {0.9} +test expr-old-32.12 {math functions in expressions} { + format %.6g [expr hypot(3.0, 4.0)] +} {5} +test expr-old-32.13 {math functions in expressions} { + format %.6g [expr log(2.8)] +} {1.02962} +test expr-old-32.14 {math functions in expressions} { + format %.6g [expr log10(2.8)] +} {0.447158} +test expr-old-32.15 {math functions in expressions} { + format %.6g [expr pow(2.1, 3.1)] +} {9.97424} +test expr-old-32.16 {math functions in expressions} { + format %.6g [expr sin(.1)] +} {0.0998334} +test expr-old-32.17 {math functions in expressions} { + format %.6g [expr sinh(.1)] +} {0.100167} +test expr-old-32.18 {math functions in expressions} { + format %.6g [expr sqrt(2.0)] +} {1.41421} +test expr-old-32.19 {math functions in expressions} { + format %.6g [expr tan(0.8)] +} {1.02964} +test expr-old-32.20 {math functions in expressions} { + format %.6g [expr tanh(0.8)] +} {0.664037} +} +test expr-old-32.21 {math functions in expressions} { + format %.6g [expr abs(-1.8)] +} {1.8} +test expr-old-32.22 {math functions in expressions} { + expr abs(10.0) +} {10.0} +test expr-old-32.23 {math functions in expressions} { + format %.6g [expr abs(-4)] +} {4} +test expr-old-32.24 {math functions in expressions} { + format %.6g [expr abs(66)] +} {66} + +# The following test is different for 32-bit versus 64-bit architectures. + +#test expr-old-32.25 {math functions in expressions} { +# list [catch {expr abs(0x8000000000000000)} msg] $msg +#} {1 {integer value too large to represent}} + +test expr-old-32.26 {math functions in expressions} { + expr double(1) +} {1.0} +test expr-old-32.27 {math functions in expressions} { + expr double(1.1) +} {1.1} +test expr-old-32.28 {math functions in expressions} { + expr int(1) +} {1} +test expr-old-32.29 {math functions in expressions} { + expr int(1.4) +} {1} +test expr-old-32.30 {math functions in expressions} { + expr int(1.6) +} {1} +test expr-old-32.31 {math functions in expressions} { + expr int(-1.4) +} {-1} +test expr-old-32.32 {math functions in expressions} { + expr int(-1.6) +} {-1} +#test expr-old-32.33 {math functions in expressions} { +# list [catch {expr int(1e60)} msg] $msg +#} {1 {integer value too large to represent}} +#test expr-old-32.34 {math functions in expressions} { +# list [catch {expr int(-1e60)} msg] $msg +#} {1 {integer value too large to represent}} +test expr-old-32.35 {math functions in expressions} { + expr round(1.49) +} {1} +test expr-old-32.36 {math functions in expressions} { + expr round(1.51) +} {2} +test expr-old-32.37 {math functions in expressions} { + expr round(-1.49) +} {-1} +test expr-old-32.38 {math functions in expressions} { + expr round(-1.51) +} {-2} +#test expr-old-32.39 {math functions in expressions} { +# list [catch {expr round(1e60)} msg] $msg +#} {1 {integer value too large to represent}} +#test expr-old-32.40 {math functions in expressions} { +# list [catch {expr round(-1e60)} msg] $msg +#} {1 {integer value too large to represent}} +if {0} { +test expr-old-32.41 {math functions in expressions} { + list [catch {expr pow(1.0 + 3.0 - 2, .8 * 5)} msg] $msg +} {0 16.0} +test expr-old-32.42 {math functions in expressions} { + list [catch {expr hypot(5*.8,3)} msg] $msg +} {0 5.0} +test expr-old-32.45 {math functions in expressions} { + expr (0 <= rand()) && (rand() < 1) +} {1} +test expr-old-32.46 {math functions in expressions} { + list [catch {expr rand(24)} msg] $msg +} {1 {too many arguments for math function}} +test expr-old-32.47 {math functions in expressions} { + list [catch {expr srand()} msg] $msg +} {1 {too few arguments for math function}} +test expr-old-32.48 {math functions in expressions} { + list [catch {expr srand(3.79)} msg] $msg +} {1 {can't use floating-point value as argument to srand}} +test expr-old-32.49 {math functions in expressions} { + list [catch {expr srand("")} msg] $msg +} {1 {argument to math function didn't have numeric value}} +test expr-old-32.50 {math functions in expressions} { + set result [expr round(srand(12345) * 1000)] + for {set i 0} {$i < 10} {incr i} { + lappend result [expr round(rand() * 1000)] + } + set result +} {97 834 948 36 12 51 766 585 914 784 333} +test expr-old-32.51 {math functions in expressions} { + list [catch {expr {srand([lindex "6ty" 0])}} msg] $msg +} {1 {argument to math function didn't have numeric value}} + +test expr-old-33.1 {conversions and fancy args to math functions} { + expr hypot ( 3 , 4 ) +} 5.0 +test expr-old-33.2 {conversions and fancy args to math functions} { + expr hypot ( (2.0+1.0) , 4 ) +} 5.0 +test expr-old-33.3 {conversions and fancy args to math functions} { + expr hypot ( 3 , (3.0 + 1.0) ) +} 5.0 +test expr-old-33.4 {conversions and fancy args to math functions} { + format %.6g [expr cos(acos(0.1))] +} 0.1 + +test expr-old-34.1 {errors in math functions} { + list [catch {expr func_2(1.0)} msg] $msg +} {1 {unknown math function "func_2"}} +test expr-old-34.2 {errors in math functions} { + list [catch {expr func|(1.0)} msg] $msg +} {1 {syntax error in expression "func|(1.0)"}} +test expr-old-34.3 {errors in math functions} { + list [catch {expr {hypot("a b", 2.0)}} msg] $msg +} {1 {argument to math function didn't have numeric value}} +test expr-old-34.4 {errors in math functions} { + list [catch {expr hypot(1.0 2.0)} msg] $msg +} {1 {syntax error in expression "hypot(1.0 2.0)"}} +test expr-old-34.5 {errors in math functions} { + list [catch {expr hypot(1.0, 2.0} msg] $msg +} {1 {syntax error in expression "hypot(1.0, 2.0"}} +test expr-old-34.6 {errors in math functions} { + list [catch {expr hypot(1.0 ,} msg] $msg +} {1 {syntax error in expression "hypot(1.0 ,"}} +test expr-old-34.7 {errors in math functions} { + list [catch {expr hypot(1.0)} msg] $msg +} {1 {too few arguments for math function}} +test expr-old-34.8 {errors in math functions} { + list [catch {expr hypot(1.0, 2.0, 3.0)} msg] $msg +} {1 {too many arguments for math function}} +test expr-old-34.9 {errors in math functions} { + list [catch {expr acos(-2.0)} msg] $msg $errorCode +} {1 {domain error: argument not in valid range} {ARITH DOMAIN {domain error: argument not in valid range}}} +test expr-old-34.10 {errors in math functions} { + list [catch {expr pow(-3, 1000001)} msg] $msg $errorCode +} {1 {floating-point value too large to represent} {ARITH OVERFLOW {floating-point value too large to represent}}} +test expr-old-34.11 {errors in math functions} { + list [catch {expr pow(3, 1000001)} msg] $msg $errorCode +} {1 {floating-point value too large to represent} {ARITH OVERFLOW {floating-point value too large to represent}}} +test expr-old-34.12 {errors in math functions} { + list [catch {expr -14.0*exp(100000)} msg] $msg $errorCode +} {1 {floating-point value too large to represent} {ARITH OVERFLOW {floating-point value too large to represent}}} +test expr-old-34.13 {errors in math functions} { + list [catch {expr int(1.0e30)} msg] $msg $errorCode +} {1 {integer value too large to represent} {ARITH IOVERFLOW {integer value too large to represent}}} +test expr-old-34.14 {errors in math functions} { + list [catch {expr int(-1.0e30)} msg] $msg $errorCode +} {1 {integer value too large to represent} {ARITH IOVERFLOW {integer value too large to represent}}} +test expr-old-34.15 {errors in math functions} { + list [catch {expr round(1.0e30)} msg] $msg $errorCode +} {1 {integer value too large to represent} {ARITH IOVERFLOW {integer value too large to represent}}} +test expr-old-34.16 {errors in math functions} { + list [catch {expr round(-1.0e30)} msg] $msg $errorCode +} {1 {integer value too large to represent} {ARITH IOVERFLOW {integer value too large to represent}}} + +test expr-old-36.1 {ExprLooksLikeInt procedure} { + list [catch {expr 0289} msg] $msg +} {1 {"0289" is an invalid octal number}} +test expr-old-36.2 {ExprLooksLikeInt procedure} { + set x 0289 + list [catch {expr {$x+1}} msg] $msg +} {1 {can't use invalid octal number as operand of "+"}} +test expr-old-36.3 {ExprLooksLikeInt procedure} { + list [catch {expr 0289.1} msg] $msg +} {0 289.1} +test expr-old-36.4 {ExprLooksLikeInt procedure} { + set x 0289.1 + list [catch {expr {$x+1}} msg] $msg +} {0 290.1} +test expr-old-36.5 {ExprLooksLikeInt procedure} { + set x { +22} + list [catch {expr {$x+1}} msg] $msg +} {0 23} +test expr-old-36.6 {ExprLooksLikeInt procedure} { + set x { -22} + list [catch {expr {$x+1}} msg] $msg +} {0 -21} +test expr-old-36.7 {ExprLooksLikeInt procedure} { + list [catch {expr nan} msg] $msg +} {1 {domain error: argument not in valid range}} +test expr-old-36.8 {ExprLooksLikeInt procedure} { + list [catch {expr 78e1} msg] $msg +} {0 780.0} +test expr-old-36.9 {ExprLooksLikeInt procedure} { + list [catch {expr 24E1} msg] $msg +} {0 240.0} +test expr-old-36.10 {ExprLooksLikeInt procedure} { + list [catch {expr 78e} msg] $msg +} {1 {syntax error in expression "78e"}} +} + +# test for [Bug #542588] +test expr-old-36.11 {ExprLooksLikeInt procedure} { + # define a "too large integer"; this one works also for 64bit arith + set x 665802003400000000000000 + list [catch {expr {$x+1}} msg] $msg +} {1 {can't use integer value too large to represent as operand of "+"}} + +# Special test for Pentium arithmetic bug of 1994: + +if {(4195835.0 - (4195835.0/3145727.0)*3145727.0) == 256.0} { + puts "Warning: this machine contains a defective Pentium processor" + puts "that performs arithmetic incorrectly. I recommend that you" + puts "call Intel customer service immediately at 1-800-628-8686" + puts "to request a replacement processor." + puts [expr {(4195835.0 - (4195835.0/3145727.0)*3145727.0)}] +} + +if {0.0 == 256.0} { + puts error +} + +testreport + + + + + + + + + + + + diff --git a/tests/expr.test b/tests/expr.test index a2d573a..b15ee36 100644 --- a/tests/expr.test +++ b/tests/expr.test @@ -50,6 +50,40 @@ test expr-1.10 "Subtraction" { expr {$b-$a} } {9} +test expr-1.11 "Short circuit evaluation" { + set a 100 + set c [expr {0 || [incr a]}] + list $a $c +} {101 1} + +test expr-1.12 "Short circuit evaluation" { + set a 100 + set c [expr {1 || [incr a]}] + list $a $c +} {100 1} + +test expr-1.13 "Short circuit evaluation" { + set a 100 + set c [expr {1 || [incr a] && [incr a]}] + list $a $c +} {100 1} + +test expr-1.14 "Rotate left" { + expr {1 <<< 5} +} {32} + +test expr-1.15 "Rotate left" { + expr {1 <<< 65} +} {2} + +test expr-1.16 "Rotate right" { + expr {1 >>> 48} +} {65536} + +test expr-1.17 "Rotate left" { + expr {1 >>> 63} +} {2} + # This crashes older jim test expr-2.1 "bogus unarymin" { expr {unarymin 1} @@ -57,6 +91,14 @@ test expr-2.1 "bogus unarymin" { test expr-2.2 "Ternary operator - missing colon" { list [catch {expr {1 ? 2 3}} msg] $msg -} {1 {Invalid expression}} +} {1 {Invalid expression: 1 ? 2 3}} + +test expr-2.3 "Ternary operator - missing third term" { + list [catch {expr {1 ? 2}} msg] $msg +} {1 {Invalid expression: 1 ? 2}} + +test expr-2.4 "Ternary operator - missing question" { + list [catch {expr {1 : 2}} msg] $msg +} {1 {Invalid expression: 1 : 2}} testreport |