diff options
Diffstat (limited to '')
-rw-r--r-- | evaluator.c | 1763 |
1 files changed, 891 insertions, 872 deletions
diff --git a/evaluator.c b/evaluator.c index a4e373a..abb3278 100644 --- a/evaluator.c +++ b/evaluator.c @@ -1,4 +1,4 @@ -/* $Id: evaluator.c,v 1.24 2005/03/30 04:57:50 reinelt Exp $ +/* $Id: evaluator.c,v 1.25 2005/05/08 04:32:44 reinelt Exp $ * * expression evaluation * @@ -23,6 +23,9 @@ * * * $Log: evaluator.c,v $ + * Revision 1.25 2005/05/08 04:32:44 reinelt + * CodingStyle added and applied + * * Revision 1.24 2005/03/30 04:57:50 reinelt * Evaluator speedup: use bsearch for finding functions and variables * @@ -192,68 +195,68 @@ #define CHUNK_SIZE 16 typedef enum { - T_UNDEF, - T_NAME, - T_NUMBER, - T_STRING, - T_OPERATOR, - T_VARIABLE, - T_FUNCTION + T_UNDEF, + T_NAME, + T_NUMBER, + T_STRING, + T_OPERATOR, + T_VARIABLE, + T_FUNCTION } TOKEN; typedef enum { - O_UNDEF, /* undefined */ - O_LST, /* expression lists */ - O_SET, /* variable assignements */ - O_CND, /* conditional a?b:c */ - O_COL, /* colon in a?b:c */ - O_OR, /* logical OR */ - O_AND, /* logical AND */ - O_EQ, /* equal */ - O_NE, /* not equal */ - O_LT, /* less than */ - O_LE, /* less or equal */ - O_GT, /* greater than */ - O_GE, /* greater or equal */ - O_ADD, /* addition */ - O_SUB, /* subtraction */ - O_SGN, /* sign '-' */ - O_CAT, /* string concatenation */ - O_MUL, /* multiplication */ - O_DIV, /* division */ - O_MOD, /* modulo */ - O_POW, /* power */ - O_NOT, /* logical NOT */ - O_BRO, /* open brace */ - O_COM, /* comma (argument seperator) */ - O_BRC /* closing brace */ + O_UNDEF, /* undefined */ + O_LST, /* expression lists */ + O_SET, /* variable assignements */ + O_CND, /* conditional a?b:c */ + O_COL, /* colon in a?b:c */ + O_OR, /* logical OR */ + O_AND, /* logical AND */ + O_EQ, /* equal */ + O_NE, /* not equal */ + O_LT, /* less than */ + O_LE, /* less or equal */ + O_GT, /* greater than */ + O_GE, /* greater or equal */ + O_ADD, /* addition */ + O_SUB, /* subtraction */ + O_SGN, /* sign '-' */ + O_CAT, /* string concatenation */ + O_MUL, /* multiplication */ + O_DIV, /* division */ + O_MOD, /* modulo */ + O_POW, /* power */ + O_NOT, /* logical NOT */ + O_BRO, /* open brace */ + O_COM, /* comma (argument seperator) */ + O_BRC /* closing brace */ } OPERATOR; typedef struct { - char *pattern; - int len; - OPERATOR op; + char *pattern; + int len; + OPERATOR op; } PATTERN; typedef struct { - char *name; - RESULT *value; + char *name; + RESULT *value; } VARIABLE; typedef struct { - char *name; - int argc; - void (*func)(); + char *name; + int argc; + void (*func) (); } FUNCTION; typedef struct _NODE { - TOKEN Token; - OPERATOR Operator; - RESULT *Result; - VARIABLE *Variable; - FUNCTION *Function; - int Children; - struct _NODE **Child; + TOKEN Token; + OPERATOR Operator; + RESULT *Result; + VARIABLE *Variable; + FUNCTION *Function; + int Children; + struct _NODE **Child; } NODE; @@ -261,324 +264,327 @@ typedef struct _NODE { /* operators */ /* IMPORTANT! list must be sorted by length! */ static PATTERN Pattern[] = { - { ";", 1, O_LST }, /* expression lists */ - { "=", 1, O_SET }, /* variable assignements */ - { "?", 1, O_CND }, /* conditional a?b:c */ - { ":", 1, O_COL }, /* colon a?b:c */ - { "|", 1, O_OR }, /* logical OR */ - { "&", 1, O_AND }, /* logical AND */ - { "<", 1, O_LT }, /* less than */ - { ">", 1, O_GT }, /* greater than */ - { "+", 1, O_ADD }, /* addition */ - { "-", 1, O_SUB }, /* subtraction or sign */ - { ".", 1, O_CAT }, /* string concatenation */ - { "*", 1, O_MUL }, /* multiplication */ - { "/", 1, O_DIV }, /* division */ - { "%", 1, O_MOD }, /* modulo */ - { "^", 1, O_POW }, /* power */ - { "!", 1, O_NOT }, /* logical NOT */ - { "(", 1, O_BRO }, /* open brace */ - { ",", 1, O_COM }, /* comma (argument seperator) */ - { ")", 1, O_BRC }, /* closing brace */ - { "==", 2, O_EQ }, /* equal */ - { "!=", 2, O_NE }, /* not equal */ - { "<=", 2, O_LE }, /* less or equal */ - { ">=", 2, O_GE } /* greater or equal */ + {";", 1, O_LST}, /* expression lists */ + {"=", 1, O_SET}, /* variable assignements */ + {"?", 1, O_CND}, /* conditional a?b:c */ + {":", 1, O_COL}, /* colon a?b:c */ + {"|", 1, O_OR}, /* logical OR */ + {"&", 1, O_AND}, /* logical AND */ + {"<", 1, O_LT}, /* less than */ + {">", 1, O_GT}, /* greater than */ + {"+", 1, O_ADD}, /* addition */ + {"-", 1, O_SUB}, /* subtraction or sign */ + {".", 1, O_CAT}, /* string concatenation */ + {"*", 1, O_MUL}, /* multiplication */ + {"/", 1, O_DIV}, /* division */ + {"%", 1, O_MOD}, /* modulo */ + {"^", 1, O_POW}, /* power */ + {"!", 1, O_NOT}, /* logical NOT */ + {"(", 1, O_BRO}, /* open brace */ + {",", 1, O_COM}, /* comma (argument seperator) */ + {")", 1, O_BRC}, /* closing brace */ + {"==", 2, O_EQ}, /* equal */ + {"!=", 2, O_NE}, /* not equal */ + {"<=", 2, O_LE}, /* less or equal */ + {">=", 2, O_GE} /* greater or equal */ }; -static char *Expression = NULL; -static char *ExprPtr = NULL; -static char *Word = NULL; -static TOKEN Token = T_UNDEF; +static char *Expression = NULL; +static char *ExprPtr = NULL; +static char *Word = NULL; +static TOKEN Token = T_UNDEF; static OPERATOR Operator = O_UNDEF; static VARIABLE *Variable = NULL; -static int nVariable = 0; +static int nVariable = 0; static FUNCTION *Function = NULL; -static int nFunction = 0; +static int nFunction = 0; -void DelResult (RESULT *result) +void DelResult(RESULT * result) { - result->type = 0; - result->number = 0.0; - result->length = -1; - if (result->string) { - free (result->string); - result->string = NULL; - } + result->type = 0; + result->number = 0.0; + result->length = -1; + if (result->string) { + free(result->string); + result->string = NULL; + } } -static void FreeResult (RESULT *result) +static void FreeResult(RESULT * result) { - if (result != NULL) { - DelResult(result); - free (result); - } + if (result != NULL) { + DelResult(result); + free(result); + } } -static RESULT* NewResult (void) +static RESULT *NewResult(void) { - RESULT *result = malloc(sizeof(RESULT)); - if (result == NULL) { - error ("Evaluator: cannot allocate result: out of memory!"); - return NULL; - } - result->type = 0; - result->number = 0.0; - result->length = -1; - result->string = NULL; - - return result; + RESULT *result = malloc(sizeof(RESULT)); + if (result == NULL) { + error("Evaluator: cannot allocate result: out of memory!"); + return NULL; + } + result->type = 0; + result->number = 0.0; + result->length = -1; + result->string = NULL; + + return result; } -static RESULT* DupResult (RESULT *result) +static RESULT *DupResult(RESULT * result) { - RESULT *result2; - - if ((result2 = NewResult()) == NULL) return NULL; - - result2->type = result->type; - result2->number = result->number; - - if (result->length >= 0) { - result2->length = result->length; - result2->string = malloc(result2->length); - strcpy(result2->string, result->string); - } else { - result2->length = -1; - result2->string = NULL; - } - - return result2; + RESULT *result2; + + if ((result2 = NewResult()) == NULL) + return NULL; + + result2->type = result->type; + result2->number = result->number; + + if (result->length >= 0) { + result2->length = result->length; + result2->string = malloc(result2->length); + strcpy(result2->string, result->string); + } else { + result2->length = -1; + result2->string = NULL; + } + + return result2; } -RESULT* SetResult (RESULT **result, const int type, const void *value) +RESULT *SetResult(RESULT ** result, const int type, const void *value) { - if (*result == NULL) { - if ((*result = NewResult()) == NULL) - return NULL; - } else if (type == R_NUMBER) { - DelResult(*result); - } - - if (type == R_NUMBER) { - (*result)->type = R_NUMBER; - (*result)->number = *(double*)value; - (*result)->length = -1; - (*result)->string = NULL; - } - - else if (type == R_STRING) { - int len = strlen((char*)value); - (*result)->type = R_STRING; - (*result)->number = 0.0; - if ((*result)->string == NULL || len > (*result)->length) { - /* buffer is either empty or too small */ - if ((*result)->string) free((*result)->string); - /* allocate memory in multiples of CHUNK_SIZE */ - /* note that length does not count the trailing \0 */ - (*result)->length = CHUNK_SIZE*(len/CHUNK_SIZE+1)-1; - (*result)->string = malloc((*result)->length+1); + if (*result == NULL) { + if ((*result = NewResult()) == NULL) + return NULL; + } else if (type == R_NUMBER) { + DelResult(*result); } - strcpy((*result)->string, value); - } else { - error ("Evaluator: internal error: invalid result type %d", type); - return NULL; - } - - return *result; + + if (type == R_NUMBER) { + (*result)->type = R_NUMBER; + (*result)->number = *(double *) value; + (*result)->length = -1; + (*result)->string = NULL; + } + + else if (type == R_STRING) { + int len = strlen((char *) value); + (*result)->type = R_STRING; + (*result)->number = 0.0; + if ((*result)->string == NULL || len > (*result)->length) { + /* buffer is either empty or too small */ + if ((*result)->string) + free((*result)->string); + /* allocate memory in multiples of CHUNK_SIZE */ + /* note that length does not count the trailing \0 */ + (*result)->length = CHUNK_SIZE * (len / CHUNK_SIZE + 1) - 1; + (*result)->string = malloc((*result)->length + 1); + } + strcpy((*result)->string, value); + } else { + error("Evaluator: internal error: invalid result type %d", type); + return NULL; + } + + return *result; } -double R2N (RESULT *result) +double R2N(RESULT * result) { - if (result == NULL) { - error ("Evaluator: internal error: NULL result"); + if (result == NULL) { + error("Evaluator: internal error: NULL result"); + return 0.0; + } + + if (result->type & R_NUMBER) { + return result->number; + } + + if (result->type & R_STRING) { + result->type |= R_NUMBER; + result->number = atof(result->string); + return result->number; + } + + error("Evaluator: internal error: invalid result type %d", result->type); return 0.0; - } - - if (result->type & R_NUMBER) { - return result->number; - } - - if (result->type & R_STRING) { - result->type |= R_NUMBER; - result->number = atof(result->string); - return result->number; - } - - error ("Evaluator: internal error: invalid result type %d", result->type); - return 0.0; } -char* R2S (RESULT *result) +char *R2S(RESULT * result) { - if (result==NULL) { - error ("Evaluator: internal error: NULL result"); + if (result == NULL) { + error("Evaluator: internal error: NULL result"); + return NULL; + } + + if (result->type & R_STRING) { + return result->string; + } + + if (result->type & R_NUMBER) { + result->type |= R_STRING; + if (result->string) + free(result->string); + result->length = CHUNK_SIZE - 1; + result->string = malloc(CHUNK_SIZE); + snprintf(result->string, CHUNK_SIZE, "%g", result->number); + return result->string; + } + + error("Evaluator: internal error: invalid result type %d", result->type); return NULL; - } - - if (result->type & R_STRING) { - return result->string; - } - - if (result->type & R_NUMBER) { - result->type |= R_STRING; - if (result->string) free(result->string); - result->length = CHUNK_SIZE-1; - result->string = malloc(CHUNK_SIZE); - snprintf(result->string, CHUNK_SIZE, "%g", result->number); - return result->string; - } - - error ("Evaluator: internal error: invalid result type %d", result->type); - return NULL; - + } /* bsearch compare function for variables */ -static int LookupVariable (const void *a, const void *b) +static int LookupVariable(const void *a, const void *b) { - char *key = (char*)a; - VARIABLE *var = (VARIABLE*)b; + char *key = (char *) a; + VARIABLE *var = (VARIABLE *) b; - return strcmp(key, var->name); + return strcmp(key, var->name); } /* qsort compare function for variables */ -static int SortVariable (const void *a, const void *b) +static int SortVariable(const void *a, const void *b) { - VARIABLE *va=(VARIABLE*)a; - VARIABLE *vb=(VARIABLE*)b; + VARIABLE *va = (VARIABLE *) a; + VARIABLE *vb = (VARIABLE *) b; - return strcmp(va->name, vb->name); + return strcmp(va->name, vb->name); } -static VARIABLE *FindVariable (const char *name) +static VARIABLE *FindVariable(const char *name) { - return bsearch(name, Variable, nVariable, sizeof(VARIABLE), LookupVariable); + return bsearch(name, Variable, nVariable, sizeof(VARIABLE), LookupVariable); } -int SetVariable (const char *name, RESULT *value) +int SetVariable(const char *name, RESULT * value) { - VARIABLE *V; - - V = FindVariable(name); - if (V != NULL) { - FreeResult (V->value); - V->value = DupResult(value); - return 1; - } - - nVariable++; - Variable = realloc(Variable, nVariable*sizeof(VARIABLE)); - Variable[nVariable-1].name = strdup(name); - Variable[nVariable-1].value = DupResult(value); - - qsort(Variable, nVariable, sizeof(VARIABLE), SortVariable); - - return 0; + VARIABLE *V; + + V = FindVariable(name); + if (V != NULL) { + FreeResult(V->value); + V->value = DupResult(value); + return 1; + } + + nVariable++; + Variable = realloc(Variable, nVariable * sizeof(VARIABLE)); + Variable[nVariable - 1].name = strdup(name); + Variable[nVariable - 1].value = DupResult(value); + + qsort(Variable, nVariable, sizeof(VARIABLE), SortVariable); + + return 0; } -int SetVariableNumeric (const char *name, const double value) +int SetVariableNumeric(const char *name, const double value) { - RESULT result = {0, 0, 0, NULL}; - RESULT *rp = &result; - - SetResult (&rp, R_NUMBER, &value); + RESULT result = { 0, 0, 0, NULL }; + RESULT *rp = &result; - return SetVariable (name, rp); + SetResult(&rp, R_NUMBER, &value); + + return SetVariable(name, rp); } -int SetVariableString (const char *name, const char *value) +int SetVariableString(const char *name, const char *value) { - RESULT result = {0, 0, 0, NULL}; - RESULT *rp = &result; + RESULT result = { 0, 0, 0, NULL }; + RESULT *rp = &result; - SetResult(&rp, R_STRING, value); + SetResult(&rp, R_STRING, value); - return SetVariable (name, rp); + return SetVariable(name, rp); } -void DeleteVariables(void) +void DeleteVariables(void) { - int i; - - for (i=0;i<nVariable;i++) { - free(Variable[i].name); - FreeResult(Variable[i].value); - } - free(Variable); - Variable = NULL; - nVariable = 0; + int i; + + for (i = 0; i < nVariable; i++) { + free(Variable[i].name); + FreeResult(Variable[i].value); + } + free(Variable); + Variable = NULL; + nVariable = 0; } /* bsearch compare function for functions */ -static int LookupFunction (const void *a, const void *b) +static int LookupFunction(const void *a, const void *b) { - char *n = (char*)a; - FUNCTION *f = (FUNCTION*)b; + char *n = (char *) a; + FUNCTION *f = (FUNCTION *) b; - return strcmp(n, f->name); + return strcmp(n, f->name); } /* qsort compare function for functions */ -static int SortFunction (const void *a, const void *b) +static int SortFunction(const void *a, const void *b) { - FUNCTION *fa=(FUNCTION*)a; - FUNCTION *fb=(FUNCTION*)b; + FUNCTION *fa = (FUNCTION *) a; + FUNCTION *fb = (FUNCTION *) b; - return strcmp(fa->name, fb->name); + return strcmp(fa->name, fb->name); } -static FUNCTION* FindFunction (const char *name) +static FUNCTION *FindFunction(const char *name) { - return bsearch(name, Function, nFunction, sizeof(FUNCTION), LookupFunction); + return bsearch(name, Function, nFunction, sizeof(FUNCTION), LookupFunction); } -int AddFunction (const char *name, const int argc, void (*func)()) +int AddFunction(const char *name, const int argc, void (*func) ()) { - nFunction++; - Function = realloc(Function, nFunction*sizeof(FUNCTION)); - Function[nFunction-1].name = strdup(name); - Function[nFunction-1].argc = argc; - Function[nFunction-1].func = func; - - qsort(Function, nFunction, sizeof(FUNCTION), SortFunction); - - return 0; + nFunction++; + Function = realloc(Function, nFunction * sizeof(FUNCTION)); + Function[nFunction - 1].name = strdup(name); + Function[nFunction - 1].argc = argc; + Function[nFunction - 1].func = func; + + qsort(Function, nFunction, sizeof(FUNCTION), SortFunction); + + return 0; } -void DeleteFunctions(void) +void DeleteFunctions(void) { - int i; - - for (i=0;i<nFunction;i++) { - free(Function[i].name); - } - free(Function); - Function=NULL; - nFunction=0; + int i; + + for (i = 0; i < nFunction; i++) { + free(Function[i].name); + } + free(Function); + Function = NULL; + nFunction = 0; } @@ -587,723 +593,736 @@ void DeleteFunctions(void) #define is_alpha(c) (((c) >= 'A' && (c) <= 'Z') || ((c) >= 'a' && (c) <= 'z') || ((c) == '_')) #define is_alnum(c) (is_alpha(c) || is_digit(c)) -static void Parse (void) +static void Parse(void) { - Token = T_UNDEF; - Operator = O_UNDEF; + Token = T_UNDEF; + Operator = O_UNDEF; - if (Word) { - free (Word); - Word = NULL; - } - - /* NULL expression? */ - if (ExprPtr == NULL) { - Word = strdup(""); - return; - } - - /* skip leading whitespace */ - while (is_space(*ExprPtr)) ExprPtr++; - - /* names */ - if (is_alpha(*ExprPtr)) { - char *start = ExprPtr; - while (is_alnum(*ExprPtr)) ExprPtr++; - if (*ExprPtr==':' && *(ExprPtr+1)==':' && is_alpha(*(ExprPtr+2))) { - ExprPtr+=3; - while (is_alnum(*ExprPtr)) ExprPtr++; + if (Word) { + free(Word); + Word = NULL; + } + + /* NULL expression? */ + if (ExprPtr == NULL) { + Word = strdup(""); + return; } - Word = strndup(start, ExprPtr-start); - Token = T_NAME; - } - - /* numbers */ - else if (is_digit(*ExprPtr) || (*ExprPtr=='.' && is_digit(*(ExprPtr+1)))) { - char *start = ExprPtr; - while (is_digit(*ExprPtr)) ExprPtr++; - if (*ExprPtr=='.') { - ExprPtr++; - while (is_digit(*ExprPtr)) ExprPtr++; + + /* skip leading whitespace */ + while (is_space(*ExprPtr)) + ExprPtr++; + + /* names */ + if (is_alpha(*ExprPtr)) { + char *start = ExprPtr; + while (is_alnum(*ExprPtr)) + ExprPtr++; + if (*ExprPtr == ':' && *(ExprPtr + 1) == ':' && is_alpha(*(ExprPtr + 2))) { + ExprPtr += 3; + while (is_alnum(*ExprPtr)) + ExprPtr++; + } + Word = strndup(start, ExprPtr - start); + Token = T_NAME; } - Word = strndup(start, ExprPtr-start); - Token = T_NUMBER; - } - - /* strings */ - else if (*ExprPtr=='\'') { - char *start=++ExprPtr; - while (*ExprPtr!='\0' && *ExprPtr!='\'') ExprPtr++; - Word = strndup(start, ExprPtr-start); - Token = T_STRING; - if (*ExprPtr=='\'') ExprPtr++; - } - - /* operators */ - else { - int i; - for (i=sizeof(Pattern)/sizeof(Pattern[0])-1; i>=0; i--) { - int len=Pattern[i].len; - if (strncmp (ExprPtr, Pattern[i].pattern, Pattern[i].len)==0) { - Word = strndup(ExprPtr, len); - Token = T_OPERATOR; - Operator = Pattern[i].op; - ExprPtr += len; - break; - } + + /* numbers */ + else if (is_digit(*ExprPtr) || (*ExprPtr == '.' && is_digit(*(ExprPtr + 1)))) { + char *start = ExprPtr; + while (is_digit(*ExprPtr)) + ExprPtr++; + if (*ExprPtr == '.') { + ExprPtr++; + while (is_digit(*ExprPtr)) + ExprPtr++; + } + Word = strndup(start, ExprPtr - start); + Token = T_NUMBER; } - } - - /* syntax check */ - if (Token == T_UNDEF && *ExprPtr != '\0') { - error ("Evaluator: parse error in <%s>: garbage <%s>", Expression, ExprPtr); - } - - /* skip trailing whitespace */ - while (is_space(*ExprPtr)) ExprPtr++; - - /* empty token */ - if (Word==NULL) Word=strdup(""); + + /* strings */ + else if (*ExprPtr == '\'') { + char *start = ++ExprPtr; + while (*ExprPtr != '\0' && *ExprPtr != '\'') + ExprPtr++; + Word = strndup(start, ExprPtr - start); + Token = T_STRING; + if (*ExprPtr == '\'') + ExprPtr++; + } + + /* operators */ + else { + int i; + for (i = sizeof(Pattern) / sizeof(Pattern[0]) - 1; i >= 0; i--) { + int len = Pattern[i].len; + if (strncmp(ExprPtr, Pattern[i].pattern, Pattern[i].len) == 0) { + Word = strndup(ExprPtr, len); + Token = T_OPERATOR; + Operator = Pattern[i].op; + ExprPtr += len; + break; + } + } + } + + /* syntax check */ + if (Token == T_UNDEF && *ExprPtr != '\0') { + error("Evaluator: parse error in <%s>: garbage <%s>", Expression, ExprPtr); + } + + /* skip trailing whitespace */ + while (is_space(*ExprPtr)) + ExprPtr++; + + /* empty token */ + if (Word == NULL) + Word = strdup(""); } -static NODE* NewNode (NODE *Child) +static NODE *NewNode(NODE * Child) { - NODE *N; - - N=malloc(sizeof(NODE)); - if (N==NULL) return NULL; - - memset (N, 0, sizeof(NODE)); - N->Token = Token; - N->Operator = Operator; - - if (Child != NULL) { - N->Children = 1; - N->Child = malloc(sizeof(NODE*)); - N->Child[0] = Child; - } - - return N; - + NODE *N; + + N = malloc(sizeof(NODE)); + if (N == NULL) + return NULL; + + memset(N, 0, sizeof(NODE)); + N->Token = Token; + N->Operator = Operator; + + if (Child != NULL) { + N->Children = 1; + N->Child = malloc(sizeof(NODE *)); + N->Child[0] = Child; + } + + return N; + } -static NODE* JunkNode (void) +static NODE *JunkNode(void) { - NODE *Junk; - - Junk = NewNode(NULL); - Junk->Token = T_STRING; - SetResult (&Junk->Result, R_STRING, ""); - - return Junk; + NODE *Junk; + + Junk = NewNode(NULL); + Junk->Token = T_STRING; + SetResult(&Junk->Result, R_STRING, ""); + + return Junk; } -static void LinkNode (NODE *Root, NODE *Child) +static void LinkNode(NODE * Root, NODE * Child) { - - if (Child == NULL) return; - Root->Children++; - Root->Child = realloc (Root->Child, Root->Children*sizeof(NODE*)); - if (Root->Child==NULL) return; - Root->Child[Root->Children-1]=Child; + if (Child == NULL) + return; + + Root->Children++; + Root->Child = realloc(Root->Child, Root->Children * sizeof(NODE *)); + if (Root->Child == NULL) + return; + Root->Child[Root->Children - 1] = Child; } /* forward declaration */ -static NODE* Level01 (void); +static NODE *Level01(void); /* literal numbers, variables, functions */ -static NODE* Level12 (void) +static NODE *Level12(void) { - NODE *Root = NULL; - - if (Token == T_OPERATOR && Operator == O_BRO) { - Parse(); - Root = Level01(); - if (Token != T_OPERATOR || Operator != O_BRC) { - error ("Evaluator: unbalanced parentheses in <%s>", Expression); - LinkNode (Root, JunkNode()); - } - } - - else if (Token == T_NUMBER) { - double value = atof(Word); - Root = NewNode(NULL); - SetResult (&Root->Result, R_NUMBER, &value); - } - - else if (Token == T_STRING) { - Root = NewNode(NULL); - SetResult (&Root->Result, R_STRING, Word); - } - - else if (Token == T_NAME) { - - /* look-ahead for opening brace */ - if (*ExprPtr == '(') { - int argc=0; - Root = NewNode(NULL); - Root->Token = T_FUNCTION; - Root->Result = NewResult(); - Root->Function = FindFunction(Word); - if (Root->Function == NULL) { - error ("Evaluator: unknown function '%s' in <%s>", Word, Expression); - Root->Token = T_STRING; - SetResult (&Root->Result, R_STRING, ""); - } - - /* opening brace */ - Parse(); - do { - Parse(); /* read argument */ - if (Token == T_OPERATOR && Operator == O_BRC) { - break; - } - else if (Token == T_OPERATOR && Operator == O_COM) { - error ("Evaluator: empty argument in <%s>", Expression); - LinkNode (Root, JunkNode()); - } - else { - LinkNode (Root, Level01()); + NODE *Root = NULL; + + if (Token == T_OPERATOR && Operator == O_BRO) { + Parse(); + Root = Level01(); + if (Token != T_OPERATOR || Operator != O_BRC) { + error("Evaluator: unbalanced parentheses in <%s>", Expression); + LinkNode(Root, JunkNode()); } - argc++; - } while (Token == T_OPERATOR && Operator == O_COM); - - /* check for closing brace */ - if (Token != T_OPERATOR || Operator != O_BRC) { - error ("Evaluator: missing closing brace in <%s>", Expression); - } - - /* check number of arguments */ - if (Root->Function != NULL && Root->Function->argc >= 0 && Root->Function->argc != argc) { - error ("Evaluator: wrong number of arguments in <%s>", Expression); - while (argc < Root->Function->argc) { - LinkNode (Root, JunkNode()); - argc++; + } + + else if (Token == T_NUMBER) { + double value = atof(Word); + Root = NewNode(NULL); + SetResult(&Root->Result, R_NUMBER, &value); + } + + else if (Token == T_STRING) { + Root = NewNode(NULL); + SetResult(&Root->Result, R_STRING, Word); + } + + else if (Token == T_NAME) { + + /* look-ahead for opening brace */ + if (*ExprPtr == '(') { + int argc = 0; + Root = NewNode(NULL); + Root->Token = T_FUNCTION; + Root->Result = NewResult(); + Root->Function = FindFunction(Word); + if (Root->Function == NULL) { + error("Evaluator: unknown function '%s' in <%s>", Word, Expression); + Root->Token = T_STRING; + SetResult(&Root->Result, R_STRING, ""); + } + + /* opening brace */ + Parse(); + do { + Parse(); /* read argument */ + if (Token == T_OPERATOR && Operator == O_BRC) { + break; + } else if (Token == T_OPERATOR && Operator == O_COM) { + error("Evaluator: empty argument in <%s>", Expression); + LinkNode(Root, JunkNode()); + } else { + LinkNode(Root, Level01()); + } + argc++; + } while (Token == T_OPERATOR && Operator == O_COM); + + /* check for closing brace */ + if (Token != T_OPERATOR || Operator != O_BRC) { + error("Evaluator: missing closing brace in <%s>", Expression); + } + + /* check number of arguments */ + if (Root->Function != NULL && Root->Function->argc >= 0 && Root->Function->argc != argc) { + error("Evaluator: wrong number of arguments in <%s>", Expression); + while (argc < Root->Function->argc) { + LinkNode(Root, JunkNode()); + argc++; + } + } + + } else { + Root = NewNode(NULL); + Root->Token = T_VARIABLE; + Root->Result = NewResult(); + Root->Variable = FindVariable(Word); + if (Root->Variable == NULL) { + SetVariableString(Word, ""); + Root->Variable = FindVariable(Word); + } } - } - - } else { - Root = NewNode(NULL); - Root->Token = T_VARIABLE; - Root->Result = NewResult(); - Root->Variable = FindVariable(Word); - if (Root->Variable == NULL) { - SetVariableString (Word, ""); - Root->Variable = FindVariable(Word); - } } - } - - else { - error ("Evaluator: syntax error in <%s>: <%s>", Expression, Word); - Root = NewNode(NULL); - Root->Token = T_STRING; - SetResult (&Root->Result, R_STRING, ""); - } - - Parse(); - return Root; - + + else { + error("Evaluator: syntax error in <%s>: <%s>", Expression, Word); + Root = NewNode(NULL); + Root->Token = T_STRING; + SetResult(&Root->Result, R_STRING, ""); + } + + Parse(); + return Root; + } /* unary + or - signs or logical 'not' */ -static NODE* Level11 (void) +static NODE *Level11(void) { - NODE *Root; - TOKEN sign = T_UNDEF; - - if (Token == T_OPERATOR && (Operator == O_ADD || Operator == O_SUB || Operator == O_NOT)) { - sign = Operator; - if (sign == O_SUB) sign = O_SGN; - Parse(); - } - - Root = Level12(); - - if (sign == O_SGN || sign == O_NOT) { - Root = NewNode (Root); - Root->Token = T_OPERATOR; - Root->Operator = sign; - } - - return Root; + NODE *Root; + TOKEN sign = T_UNDEF; + + if (Token == T_OPERATOR && (Operator == O_ADD || Operator == O_SUB || Operator == O_NOT)) { + sign = Operator; + if (sign == O_SUB) + sign = O_SGN; + Parse(); + } + + Root = Level12(); + + if (sign == O_SGN || sign == O_NOT) { + Root = NewNode(Root); + Root->Token = T_OPERATOR; + Root->Operator = sign; + } + + return Root; } /* x^y */ -static NODE* Level10 (void) +static NODE *Level10(void) { - NODE *Root; + NODE *Root; - Root = Level11(); - - while (Token == T_OPERATOR && Operator == O_POW) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level11()); - } - - return Root; + Root = Level11(); + + while (Token == T_OPERATOR && Operator == O_POW) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level11()); + } + + return Root; } /* multiplication, division, modulo */ -static NODE* Level09 (void) +static NODE *Level09(void) { - NODE *Root; + NODE *Root; - Root = Level10(); - - while (Token == T_OPERATOR && (Operator == O_MUL || Operator == O_DIV || Operator == O_MOD)) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level10()); - } - - return Root; + Root = Level10(); + + while (Token == T_OPERATOR && (Operator == O_MUL || Operator == O_DIV || Operator == O_MOD)) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level10()); + } + + return Root; } /* addition, subtraction, string concatenation */ -static NODE* Level08 (void) +static NODE *Level08(void) { - NODE *Root; + NODE *Root; - Root = Level09(); - - while (Token == T_OPERATOR && (Operator == O_ADD || Operator == O_SUB || Operator == O_CAT)) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level09()); - } - - return Root; + Root = Level09(); + + while (Token == T_OPERATOR && (Operator == O_ADD || Operator == O_SUB || Operator == O_CAT)) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level09()); + } + + return Root; } /* relational operators */ -static NODE* Level07 (void) +static NODE *Level07(void) { - NODE *Root; + NODE *Root; - Root = Level08(); - - while (Token == T_OPERATOR && (Operator == O_GT || Operator == O_GE || Operator == O_LT || Operator == O_LE)) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level08()); - } - - return Root; + Root = Level08(); + + while (Token == T_OPERATOR && (Operator == O_GT || Operator == O_GE || Operator == O_LT || Operator == O_LE)) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level08()); + } + + return Root; } /* equal, not equal */ -static NODE* Level06 (void) +static NODE *Level06(void) { - NODE *Root; + NODE *Root; - Root = Level07(); - - while (Token == T_OPERATOR && (Operator == O_EQ || Operator == O_NE)) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level07()); - } - - return Root; + Root = Level07(); + + while (Token == T_OPERATOR && (Operator == O_EQ || Operator == O_NE)) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level07()); + } + + return Root; } /* logical 'and' */ -static NODE* Level05 (void) +static NODE *Level05(void) { - NODE *Root; + NODE *Root; - Root = Level06(); - - while (Token == T_OPERATOR && Operator == O_AND) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level06()); - } - - return Root; + Root = Level06(); + + while (Token == T_OPERATOR && Operator == O_AND) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level06()); + } + + return Root; } /* logical 'or' */ -static NODE* Level04 (void) +static NODE *Level04(void) { - NODE *Root; + NODE *Root; - Root = Level05(); - - while (Token == T_OPERATOR && Operator == O_OR) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level05()); - } - - return Root; + Root = Level05(); + + while (Token == T_OPERATOR && Operator == O_OR) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level05()); + } + + return Root; } /* conditional expression a?b:c */ -static NODE* Level03 (void) +static NODE *Level03(void) { - NODE *Root; - - Root = Level04(); - - if (Token == T_OPERATOR && Operator == O_CND) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level04()); - if (Token == T_OPERATOR && Operator == O_COL) { - Parse(); - LinkNode (Root, Level04()); - } else { - error ("Evaluator: syntax error in <%s>: expecting ':' got '%s'", Expression, Word); - LinkNode (Root, JunkNode()); + NODE *Root; + + Root = Level04(); + + if (Token == T_OPERATOR && Operator == O_CND) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level04()); + if (Token == T_OPERATOR && Operator == O_COL) { + Parse(); + LinkNode(Root, Level04()); + } else { + error("Evaluator: syntax error in <%s>: expecting ':' got '%s'", Expression, Word); + LinkNode(Root, JunkNode()); + } } - } - return Root; + return Root; } /* variable assignments */ -static NODE* Level02 (void) +static NODE *Level02(void) { - NODE *Root; - - /* we have to do a look-ahead if it's really an assignment */ - if ((Token == T_NAME) && (*ExprPtr == '=') && (*(ExprPtr+1) != '=')) { - char *name = strdup(Word); - VARIABLE *V = FindVariable (name); - if (V == NULL) { - SetVariableString (name, ""); - V = FindVariable (name); + NODE *Root; + + /* we have to do a look-ahead if it's really an assignment */ + if ((Token == T_NAME) && (*ExprPtr == '=') && (*(ExprPtr + 1) != '=')) { + char *name = strdup(Word); + VARIABLE *V = FindVariable(name); + if (V == NULL) { + SetVariableString(name, ""); + V = FindVariable(name); + } + Parse(); + Root = NewNode(NULL); + Root->Variable = V; + Parse(); + LinkNode(Root, Level03()); + free(name); + } else { + Root = Level03(); } - Parse(); - Root = NewNode (NULL); - Root->Variable = V; - Parse(); - LinkNode (Root, Level03()); - free (name); - } else { - Root = Level03(); - } - - return Root; + + return Root; } /* expression lists */ -static NODE* Level01 (void) +static NODE *Level01(void) { - NODE *Root; + NODE *Root; - Root = Level02(); - - while (Token == T_OPERATOR && Operator == O_LST) { - Root = NewNode (Root); - Parse(); - LinkNode (Root, Level02()); - } - - return Root; + Root = Level02(); + + while (Token == T_OPERATOR && Operator == O_LST) { + Root = NewNode(Root); + Parse(); + LinkNode(Root, Level02()); + } + + return Root; } -static int EvalTree (NODE *Root) +static int EvalTree(NODE * Root) { - int i; - int argc; - int type = -1; - double number = 0.0; - double dummy; - int freeme = 0; - char *string = NULL; - char *s1, *s2; - RESULT *param[10]; - - for (i = 0; i < Root->Children; i++) { - EvalTree (Root->Child[i]); - } - - switch (Root->Token) { - - case T_NUMBER: - case T_STRING: - /* Root->Result already contains the value */ - return 0; - - case T_VARIABLE: - DelResult (Root->Result); - Root->Result = DupResult (Root->Variable->value); - return 0; - - case T_FUNCTION: - DelResult (Root->Result); - /* prepare parameter list */ - argc = Root->Children; - if (argc>10) argc=10; - for (i = 0; i < argc; i++) { - param[i]=Root->Child[i]->Result; - } - if (Root->Function->argc < 0) { - /* Function with variable argument list: */ - /* pass number of arguments as first parameter */ - Root->Function->func(Root->Result, argc, ¶m); - } else { - Root->Function->func(Root->Result, - param[0], param[1], param[2], param[3], param[4], - param[5], param[6], param[7], param[8], param[9]); + int i; + int argc; + int type = -1; + double number = 0.0; + double dummy; + int freeme = 0; + char *string = NULL; + char *s1, *s2; + RESULT *param[10]; + + for (i = 0; i < Root->Children; i++) { + EvalTree(Root->Child[i]); } - return 0; - - case T_OPERATOR: - switch (Root->Operator) { - - case O_LST: /* expression list: result is last expression */ - i = Root->Children-1; - type = Root->Child[i]->Result->type; - number = Root->Child[i]->Result->number; - string = Root->Child[i]->Result->string; - break; - - case O_SET: /* variable assignment */ - DelResult(Root->Variable->value); - Root->Variable->value = DupResult (Root->Child[0]->Result); - type = Root->Child[0]->Result->type; - number = Root->Child[0]->Result->number; - string = Root->Child[0]->Result->string; - break; - - case O_CND: /* conditional expression */ - i = 1+(R2N(Root->Child[0]->Result) == 0.0); - type = Root->Child[i]->Result->type; - number = Root->Child[i]->Result->number; - string = Root->Child[i]->Result->string; - break; - - case O_OR: /* logical OR */ - type = R_NUMBER; - number = ((R2N(Root->Child[0]->Result) != 0.0) || (R2N(Root->Child[1]->Result) != 0.0)); - break; - - case O_AND: /* logical AND */ - type = R_NUMBER; - number = ((R2N(Root->Child[0]->Result) != 0.0) && (R2N(Root->Child[1]->Result) != 0.0)); - break; - - case O_EQ: /* numeric equal */ - type = R_NUMBER; - number = (R2N(Root->Child[0]->Result) == R2N(Root->Child[1]->Result)); - break; - - case O_NE: /* numeric not equal */ - type = R_NUMBER; - number = (R2N(Root->Child[0]->Result) != R2N(Root->Child[1]->Result)); - break; - - case O_LT: /* numeric less than */ - type = R_NUMBER; - number = (R2N(Root->Child[0]->Result) < R2N(Root->Child[1]->Result)); - break; - - case O_LE: /* numeric less equal */ - type = R_NUMBER; - number = (R2N(Root->Child[0]->Result) <= R2N(Root->Child[1]->Result)); - break; - - case O_GT: /* numeric greater than */ - type = R_NUMBER; - number = (R2N(Root->Child[0]->Result) > R2N(Root->Child[1]->Result)); - break; - - case O_GE: /* numeric greater equal */ - type = R_NUMBER; - number = (R2N(Root->Child[0]->Result) >= R2N(Root->Child[1]->Result)); - break; - - case O_ADD: /* addition */ - type = R_NUMBER; - number = R2N(Root->Child[0]->Result) + R2N(Root->Child[1]->Result); - break; - - case O_SUB: /* subtraction */ - type = R_NUMBER; - number = R2N(Root->Child[0]->Result) - R2N(Root->Child[1]->Result); - break; - - case O_SGN: /* sign */ - type = R_NUMBER; - number = -R2N(Root->Child[0]->Result); - break; - - case O_CAT: /* string concatenation */ - type = R_STRING; - s1 = R2S(Root->Child[0]->Result); - s2 = R2S(Root->Child[1]->Result); - string = malloc(strlen(s1)+strlen(s2)+1); - strcpy (string, s1); - strcat (string, s2); - freeme = 1; - break; - - case O_MUL: /* multiplication */ - type = R_NUMBER; - number = R2N(Root->Child[0]->Result) * R2N(Root->Child[1]->Result); - break; - - case O_DIV: /* division */ - type = R_NUMBER; - dummy = R2N(Root->Child[1]->Result); - if (dummy == 0) { - error ("Evaluator: warning: division by zero"); - number = 0.0; - } else { - number = R2N(Root->Child[0]->Result) / R2N(Root->Child[1]->Result); - } - break; - - case O_MOD: /* modulo */ - type = R_NUMBER; - dummy = R2N(Root->Child[1]->Result); - if (dummy == 0) { - error ("Evaluator: warning: division by zero"); - number = 0.0; - } else { - number = fmod(R2N(Root->Child[0]->Result), R2N(Root->Child[1]->Result)); - } - break; - - case O_POW: /* x^y */ - type = R_NUMBER; - number = pow(R2N(Root->Child[0]->Result), R2N(Root->Child[1]->Result)); - break; - - case O_NOT: /* logical NOT */ - type = R_NUMBER; - number = (R2N(Root->Child[0]->Result) == 0.0); - break; + + switch (Root->Token) { + + case T_NUMBER: + case T_STRING: + /* Root->Result already contains the value */ + return 0; + + case T_VARIABLE: + DelResult(Root->Result); + Root->Result = DupResult(Root->Variable->value); + return 0; + + case T_FUNCTION: + DelResult(Root->Result); + /* prepare parameter list */ + argc = Root->Children; + if (argc > 10) + argc = 10; + for (i = 0; i < argc; i++) { + param[i] = Root->Child[i]->Result; + } + if (Root->Function->argc < 0) { + /* Function with variable argument list: */ + /* pass number of arguments as first parameter */ + Root->Function->func(Root->Result, argc, ¶m); + } else { + Root->Function->func(Root->Result, param[0], param[1], param[2], param[3], param[4], param[5], param[6], param[7], param[8], param[9]); + } + return 0; + + case T_OPERATOR: + switch (Root->Operator) { + + case O_LST: /* expression list: result is last expression */ + i = Root->Children - 1; + type = Root->Child[i]->Result->type; + number = Root->Child[i]->Result->number; + string = Root->Child[i]->Result->string; + break; + + case O_SET: /* variable assignment */ + DelResult(Root->Variable->value); + Root->Variable->value = DupResult(Root->Child[0]->Result); + type = Root->Child[0]->Result->type; + number = Root->Child[0]->Result->number; + string = Root->Child[0]->Result->string; + break; + + case O_CND: /* conditional expression */ + i = 1 + (R2N(Root->Child[0]->Result) == 0.0); + type = Root->Child[i]->Result->type; + number = Root->Child[i]->Result->number; + string = Root->Child[i]->Result->string; + break; + + case O_OR: /* logical OR */ + type = R_NUMBER; + number = ((R2N(Root->Child[0]->Result) != 0.0) || (R2N(Root->Child[1]->Result) != 0.0)); + break; + + case O_AND: /* logical AND */ + type = R_NUMBER; + number = ((R2N(Root->Child[0]->Result) != 0.0) && (R2N(Root->Child[1]->Result) != 0.0)); + break; + + case O_EQ: /* numeric equal */ + type = R_NUMBER; + number = (R2N(Root->Child[0]->Result) == R2N(Root->Child[1]->Result)); + break; + + case O_NE: /* numeric not equal */ + type = R_NUMBER; + number = (R2N(Root->Child[0]->Result) != R2N(Root->Child[1]->Result)); + break; + + case O_LT: /* numeric less than */ + type = R_NUMBER; + number = (R2N(Root->Child[0]->Result) < R2N(Root->Child[1]->Result)); + break; + + case O_LE: /* numeric less equal */ + type = R_NUMBER; + number = (R2N(Root->Child[0]->Result) <= R2N(Root->Child[1]->Result)); + break; + + case O_GT: /* numeric greater than */ + type = R_NUMBER; + number = (R2N(Root->Child[0]->Result) > R2N(Root->Child[1]->Result)); + break; + + case O_GE: /* numeric greater equal */ + type = R_NUMBER; + number = (R2N(Root->Child[0]->Result) >= R2N(Root->Child[1]->Result)); + break; + + case O_ADD: /* addition */ + type = R_NUMBER; + number = R2N(Root->Child[0]->Result) + R2N(Root->Child[1]->Result); + break; + + case O_SUB: /* subtraction */ + type = R_NUMBER; + number = R2N(Root->Child[0]->Result) - R2N(Root->Child[1]->Result); + break; + + case O_SGN: /* sign */ + type = R_NUMBER; + number = -R2N(Root->Child[0]->Result); + break; + + case O_CAT: /* string concatenation */ + type = R_STRING; + s1 = R2S(Root->Child[0]->Result); + s2 = R2S(Root->Child[1]->Result); + string = malloc(strlen(s1) + strlen(s2) + 1); + strcpy(string, s1); + strcat(string, s2); + freeme = 1; + break; + + case O_MUL: /* multiplication */ + type = R_NUMBER; + number = R2N(Root->Child[0]->Result) * R2N(Root->Child[1]->Result); + break; + + case O_DIV: /* division */ + type = R_NUMBER; + dummy = R2N(Root->Child[1]->Result); + if (dummy == 0) { + error("Evaluator: warning: division by zero"); + number = 0.0; + } else { + number = R2N(Root->Child[0]->Result) / R2N(Root->Child[1]->Result); + } + break; + + case O_MOD: /* modulo */ + type = R_NUMBER; + dummy = R2N(Root->Child[1]->Result); + if (dummy == 0) { + error("Evaluator: warning: division by zero"); + number = 0.0; + } else { + number = fmod(R2N(Root->Child[0]->Result), R2N(Root->Child[1]->Result)); + } + break; + + case O_POW: /* x^y */ + type = R_NUMBER; + number = pow(R2N(Root->Child[0]->Result), R2N(Root->Child[1]->Result)); + break; + + case O_NOT: /* logical NOT */ + type = R_NUMBER; + number = (R2N(Root->Child[0]->Result) == 0.0); + break; + + default: + error("Evaluator: internal error: unhandled operator <%d>", Root->Operator); + SetResult(&Root->Result, R_STRING, ""); + return -1; + } + + if (type == R_NUMBER) { + SetResult(&Root->Result, R_NUMBER, &number); + return 0; + } + if (type == R_STRING) { + SetResult(&Root->Result, R_STRING, string); + if (freeme) + free(string); + return 0; + } + error("Evaluator: internal error: unhandled type <%d>", type); + SetResult(&Root->Result, R_STRING, ""); + return -1; default: - error ("Evaluator: internal error: unhandled operator <%d>", Root->Operator); - SetResult (&Root->Result, R_STRING, ""); - return -1; - } - - if (type==R_NUMBER) { - SetResult (&Root->Result, R_NUMBER, &number); - return 0; - } - if (type==R_STRING) { - SetResult (&Root->Result, R_STRING, string); - if (freeme) free (string); - return 0; + error("Evaluator: internal error: unhandled token <%d>", Root->Token); + SetResult(&Root->Result, R_STRING, ""); + return -1; + } - error ("Evaluator: internal error: unhandled type <%d>", type); - SetResult (&Root->Result, R_STRING, ""); - return -1; - - default: - error ("Evaluator: internal error: unhandled token <%d>", Root->Token); - SetResult (&Root->Result, R_STRING, ""); - return -1; - - } - - return 0; + + return 0; } -int Compile (const char* expression, void **tree) +int Compile(const char *expression, void **tree) { - NODE *Root; - - *tree = NULL; - - Expression = (char*) expression; - ExprPtr = Expression; - - Parse(); - if (*Word=='\0') { - /* error ("Evaluator: empty expression <%s>", Expression); */ - free (Word); - Word = NULL; - return -1; - } - - Root = Level01(); - - if (*Word!='\0') { - error ("Evaluator: syntax error in <%s>: garbage <%s>", Expression, Word); - free (Word); + NODE *Root; + + *tree = NULL; + + Expression = (char *) expression; + ExprPtr = Expression; + + Parse(); + if (*Word == '\0') { + /* error ("Evaluator: empty expression <%s>", Expression); */ + free(Word); + Word = NULL; + return -1; + } + + Root = Level01(); + + if (*Word != '\0') { + error("Evaluator: syntax error in <%s>: garbage <%s>", Expression, Word); + free(Word); + Word = NULL; + return -1; + } + + free(Word); Word = NULL; - return -1; - } - - free (Word); - Word = NULL; - - *(NODE**)tree = Root; - - return 0; + + *(NODE **) tree = Root; + + return 0; } -int Eval (void *tree, RESULT *result) +int Eval(void *tree, RESULT * result) { - int ret; - NODE *Tree = (NODE*)tree; + int ret; + NODE *Tree = (NODE *) tree; - DelResult (result); - - if (Tree==NULL) { - SetResult (&result, R_STRING, ""); - return 0; - } - - ret = EvalTree(Tree); - - result->type = Tree->Result->type; - result->number = Tree->Result->number; - result->length = Tree->Result->length; - if (result->length >= 0) { - result->string = malloc(result->length); - if (Tree->Result->string != NULL) - strcpy(result->string, Tree->Result->string); - else - result->string[0]='\0'; - } else { - result->string = NULL; - } - - return ret; + DelResult(result); + + if (Tree == NULL) { + SetResult(&result, R_STRING, ""); + return 0; + } + + ret = EvalTree(Tree); + + result->type = Tree->Result->type; + result->number = Tree->Result->number; + result->length = Tree->Result->length; + if (result->length >= 0) { + result->string = malloc(result->length); + if (Tree->Result->string != NULL) + strcpy(result->string, Tree->Result->string); + else + result->string[0] = '\0'; + } else { + result->string = NULL; + } + + return ret; } -void DelTree (void *tree) +void DelTree(void *tree) { - int i; - NODE *Tree = (NODE*)tree; - - if (Tree == NULL) return; - - for (i=0; i<Tree->Children; i++) { - DelTree (Tree->Child[i]); - } - - if (Tree->Result) FreeResult (Tree->Result); - free(Tree); + int i; + NODE *Tree = (NODE *) tree; + + if (Tree == NULL) + return; + + for (i = 0; i < Tree->Children; i++) { + DelTree(Tree->Child[i]); + } + + if (Tree->Result) + FreeResult(Tree->Result); + free(Tree); } |