diff options
Diffstat (limited to 'generic/tclCompExpr.c')
-rw-r--r-- | generic/tclCompExpr.c | 559 |
1 files changed, 327 insertions, 232 deletions
diff --git a/generic/tclCompExpr.c b/generic/tclCompExpr.c index 58ddb3b..d98061c 100644 --- a/generic/tclCompExpr.c +++ b/generic/tclCompExpr.c @@ -10,28 +10,33 @@ * See the file "license.terms" for information on usage and redistribution of * this file, and for a DISCLAIMER OF ALL WARRANTIES. * - * RCS: @(#) $Id: tclCompExpr.c,v 1.45 2006/12/12 21:45:04 dgp Exp $ + * RCS: @(#) $Id: tclCompExpr.c,v 1.46 2006/12/13 16:28:06 dkf Exp $ */ #include "tclInt.h" #include "tclCompile.h" -#undef USE_EXPR_TOKENS +#undef USE_EXPR_TOKENS #undef PARSE_DIRECT_EXPR_TOKENS #ifdef PARSE_DIRECT_EXPR_TOKENS /* - * The ExprNode structure represents one node of the parse tree produced - * as an interim structure by the expression parser. + * The ExprNode structure represents one node of the parse tree produced as an + * interim structure by the expression parser. */ typedef struct ExprNode { - unsigned char lexeme; /* Code that identifies the type of this node */ - int left; /* Index of the left operand of this operator node */ - int right; /* Index of the right operand of this operator node */ - int parent; /* Index of the operator of this operand node */ - int token; /* Index of the Tcl_Tokens of this leaf node */ + unsigned char lexeme; /* Code that identifies the type of this + * node. */ + int left; /* Index of the left operand of this operator + * node. */ + int right; /* Index of the right operand of this operator + * node. */ + int parent; /* Index of the operator of this operand + * node. */ + int token; /* Index of the Tcl_Tokens of this leaf + * node. */ } ExprNode; #endif @@ -50,23 +55,26 @@ enum OperandTypes { */ typedef struct OpNode { - unsigned char lexeme; /* Code that identifies the operator */ - int left; /* Index of the left operand. Non-negative integer - is an index into the parse tree, pointing to another - operator. Value OT_LITERAL indicates operand is the - next entry in the literal list. Value OT_TOKENS - indicates the operand is the next word in the - Tcl_Parse struct. Value OT_NONE indicates we - haven't yet parsed the operand for this operator. */ - int right; /* Index of the right operand. Same interpretation - as left, with addition of OT_EMPTY meaning zero - arguments. */ - int parent; /* Index of the operator of this operand node */ + unsigned char lexeme; /* Code that identifies the operator. */ + int left; /* Index of the left operand. Non-negative + * integer is an index into the parse tree, + * pointing to another operator. Value + * OT_LITERAL indicates operand is the next + * entry in the literal list. Value OT_TOKENS + * indicates the operand is the next word in + * the Tcl_Parse struct. Value OT_NONE + * indicates we haven't yet parsed the operand + * for this operator. */ + int right; /* Index of the right operand. Same + * interpretation as left, with addition of + * OT_EMPTY meaning zero arguments. */ + int parent; /* Index of the operator of this operand + * node. */ } OpNode; /* - * Set of lexeme codes stored in ExprNode structs to label and categorize - * the lexemes found. + * Set of lexeme codes stored in ExprNode structs to label and categorize the + * lexemes found. */ #define LEAF (1<<7) @@ -131,40 +139,28 @@ typedef struct OpNode { */ static int ParseLexeme(CONST char *start, int numBytes, - unsigned char *lexemePtr, Tcl_Obj **literalPtr); - + unsigned char *lexemePtr, Tcl_Obj **literalPtr); #if (!defined(PARSE_DIRECT_EXPR_TOKENS) || !defined(USE_EXPR_TOKENS)) - static int ParseExpr(Tcl_Interp *interp, CONST char *start, - int numBytes, OpNode **opTreePtr, - Tcl_Obj *litList, Tcl_Obj *funcList, - Tcl_Parse *parsePtr); - + int numBytes, OpNode **opTreePtr, + Tcl_Obj *litList, Tcl_Obj *funcList, + Tcl_Parse *parsePtr); #endif - #ifdef PARSE_DIRECT_EXPR_TOKENS - static void GenerateTokens(ExprNode *nodes, Tcl_Parse *scratchPtr, - Tcl_Parse *parsePtr); - + Tcl_Parse *parsePtr); #else - static void ConvertTreeToTokens(Tcl_Interp *interp, - CONST char *start, int numBytes, - OpNode *nodes, Tcl_Obj *litList, - Tcl_Token *tokenPtr, Tcl_Parse *parsePtr); + CONST char *start, int numBytes, OpNode *nodes, + Tcl_Obj *litList, Tcl_Token *tokenPtr, + Tcl_Parse *parsePtr); static int GenerateTokensForLiteral(CONST char *script, - int numBytes, Tcl_Obj *litList, - int nextLiteral, Tcl_Parse *parsePtr); + int numBytes, Tcl_Obj *litList, int nextLiteral, + Tcl_Parse *parsePtr); static int CopyTokens(Tcl_Token *sourcePtr, Tcl_Parse *parsePtr); - #endif - - - - -#if (!defined(PARSE_DIRECT_EXPR_TOKENS) || !defined(USE_EXPR_TOKENS)) +#if (!defined(PARSE_DIRECT_EXPR_TOKENS) || !defined(USE_EXPR_TOKENS)) /* *---------------------------------------------------------------------- * @@ -176,11 +172,11 @@ static int CopyTokens(Tcl_Token *sourcePtr, Tcl_Parse *parsePtr); * caller. * * Results: - * If the string is successfully parsed as a valid Tcl expression, - * TCL_OK is returned, and data about the expression structure is - * written to *parsePtr. If the string cannot be parsed as a valid - * Tcl expression, TCL_ERROR is returned, and if interp is non-NULL, - * an error message is written to interp. + * If the string is successfully parsed as a valid Tcl expression, TCL_OK + * is returned, and data about the expression structure is written to + * *parsePtr. If the string cannot be parsed as a valid Tcl expression, + * TCL_ERROR is returned, and if interp is non-NULL, an error message is + * written to interp. * * Side effects: * If there is insufficient space in parsePtr to hold all the information @@ -198,10 +194,10 @@ ParseExpr( int numBytes, /* Number of bytes in string. If < 0, the * string consists of all bytes up to the * first null character. */ - OpNode **opTreePtr, /* Points to space where a pointer to - * the allocated OpNode tree should go */ - Tcl_Obj *litList, /* List to append literals to */ - Tcl_Obj *funcList, /* List to append function names to */ + OpNode **opTreePtr, /* Points to space where a pointer to the + * allocated OpNode tree should go. */ + Tcl_Obj *litList, /* List to append literals to. */ + Tcl_Obj *funcList, /* List to append function names to. */ Tcl_Parse *parsePtr) /* Structure to fill with tokens representing * those operands that require run time * substitutions. */ @@ -236,7 +232,10 @@ ParseExpr( "not enough memory to parse expression", -1); code = TCL_ERROR; } else { - /* Initialize the parse tree with the special "START" node */ + /* + * Initialize the parse tree with the special "START" node. + */ + nodes->lexeme = lexeme; nodes->left = OT_NONE; nodes->right = OT_NONE; @@ -251,16 +250,17 @@ ParseExpr( CONST char *lastStart = start - scanned; /* - * Each pass through this loop adds one more ExprNode. - * Allocate space for one if required. + * Each pass through this loop adds one more ExprNode. Allocate space + * for one if required. */ + if (nodesUsed >= nodesAvailable) { int size = nodesUsed * 2; OpNode *newPtr; do { - newPtr = (OpNode *) attemptckrealloc( (char *) nodes, - (unsigned int) (size * sizeof(OpNode)) ); + newPtr = (OpNode *) attemptckrealloc((char *) nodes, + (unsigned int) size * sizeof(OpNode)); } while ((newPtr == NULL) && ((size -= (size - nodesUsed) / 2) > nodesUsed)); if (newPtr == NULL) { @@ -274,7 +274,9 @@ ParseExpr( } nodePtr = nodes + nodesUsed; - /* Skip white space between lexemes */ + /* + * Skip white space between lexemes. + */ scanned = TclParseAllWhiteSpace(start, numBytes); start += scanned; @@ -282,7 +284,9 @@ ParseExpr( scanned = ParseLexeme(start, numBytes, &lexeme, &literal); - /* Use context to categorize the lexemes that are ambiguous */ + /* + * Use context to categorize the lexemes that are ambiguous. + */ if ((NODE_TYPE & lexeme) == 0) { switch (lexeme) { @@ -336,7 +340,9 @@ ParseExpr( } } - /* Add node to parse tree based on category */ + /* + * Add node to parse tree based on category. + */ switch (NODE_TYPE & lexeme) { case LEAF: { @@ -373,7 +379,10 @@ ParseExpr( break; } - /* Make room for at least 2 more tokens */ + /* + * Make room for at least 2 more tokens. + */ + if (parsePtr->numTokens+1 >= parsePtr->tokensAvailable) { TclExpandTokenArray(parsePtr); } @@ -399,7 +408,7 @@ ParseExpr( code = Tcl_ParseBraces(interp, start, numBytes, parsePtr, 1, &end); if (code != TCL_OK) { - continue; + continue; } scanned = end - start; break; @@ -430,8 +439,8 @@ ParseExpr( start++; while (1) { Tcl_Parse nested; - code = Tcl_ParseCommand(interp, - start, (end - start), 1, &nested); + code = Tcl_ParseCommand(interp, start, (end - start), 1, + &nested); if (code != TCL_OK) { parsePtr->term = nested.term; parsePtr->errorType = nested.errorType; @@ -509,12 +518,14 @@ ParseExpr( unsigned char precedence = prec[lexeme]; if (lastWas >= 0) { - if ((lexeme == CLOSE_PAREN) && (nodePtr[-1].lexeme == OPEN_PAREN)) { if (nodePtr[-2].lexeme == FUNCTION) { - /* Normally, "()" is a syntax error, but as a special - * case accept it as an argument list for a function */ + /* + * Normally, "()" is a syntax error, but as a special + * case accept it as an argument list for a function. + */ + scanned = 0; lastWas = OT_EMPTY; nodePtr[-1].left--; @@ -531,7 +542,7 @@ ParseExpr( if (nodePtr[-1].lexeme == OPEN_PAREN) { msg = Tcl_NewStringObj("unbalanced open paren", -1); } else if (nodePtr[-1].lexeme == COMMA) { - msg = Tcl_ObjPrintf( + msg = Tcl_ObjPrintf( "missing function argument at %s", mark); scanned = 0; insertMark = 1; @@ -566,22 +577,30 @@ ParseExpr( otherPtr = nodePtr - 1; } while (1) { - /* lastWas is "index" of item to be linked */ - /* otherPtr points to competing operator */ + /* + * lastWas is "index" of item to be linked. otherPtr points to + * competing operator. + */ if (prec[otherPtr->lexeme] < precedence) { break; } if (prec[otherPtr->lexeme] == precedence) { - /* Right association rules for exponentiation. */ + /* + * Right association rules for exponentiation. + */ + if (lexeme == EXPON) { break; } - /* Special association rules for the ternary operators. + + /* + * Special association rules for the ternary operators. * The "?" and ":" operators have equal precedence, but * must be linked up in sensible pairs. */ + if ((otherPtr->lexeme == QUESTION) && ((lastWas < 0) || (nodes[lastWas].lexeme != COLON))) { break; @@ -591,9 +610,11 @@ ParseExpr( } } - /* We should link the lastWas item to the otherPtr - * as its right operand. First make some syntax checks + /* + * We should link the lastWas item to the otherPtr as its + * right operand. First make some syntax checks. */ + if ((otherPtr->lexeme == OPEN_PAREN) && (lexeme != CLOSE_PAREN)) { msg = Tcl_NewStringObj("unbalanced open paren", -1); @@ -618,7 +639,10 @@ ParseExpr( break; } - /* Link orphan as right operand of otherPtr */ + /* + * Link orphan as right operand of otherPtr. + */ + otherPtr->right = lastWas; if (lastWas >= 0) { nodes[lastWas].parent = otherPtr - nodes; @@ -626,11 +650,17 @@ ParseExpr( lastWas = otherPtr - nodes; if (otherPtr->lexeme == OPEN_PAREN) { - /* CLOSE_PAREN can only close one OPEN_PAREN */ + /* + * CLOSE_PAREN can only close one OPEN_PAREN. + */ + break; } if (otherPtr->lexeme == START) { - /* Don't backtrack beyond the start */ + /* + * Don't backtrack beyond the start. + */ + break; } otherPtr = nodes + otherPtr->parent; @@ -648,7 +678,11 @@ ParseExpr( lastWas = OT_NONE; lastOpen = otherPtr - nodes; otherPtr->left++; - /* Create no node for a CLOSE_PAREN lexeme */ + + /* + * Create no node for a CLOSE_PAREN lexeme. + */ + break; } if (lexeme == COMMA) { @@ -670,7 +704,10 @@ ParseExpr( continue; } - /* Link orphan as left operand of new node */ + /* + * Link orphan as left operand of new node. + */ + nodePtr->lexeme = lexeme; nodePtr->right = -1; nodePtr->left = lastWas; @@ -692,50 +729,44 @@ ParseExpr( if (code == TCL_OK) { *opTreePtr = nodes; + } else if (interp == NULL) { + if (msg) { + Tcl_DecrRefCount(msg); + } } else { - if (interp == NULL) { - if (msg) { - Tcl_DecrRefCount(msg); - } - } else { - if (msg == NULL) { - msg = Tcl_GetObjResult(interp); - } - Tcl_AppendPrintfToObj(msg, - "\nin expression \"%s%.*s%.*s%s%s%.*s%s\"", - ((start - limit) < parsePtr->string) ? "" : "...", - ((start - limit) < parsePtr->string) - ? (start - parsePtr->string) : limit - 3, - ((start - limit) < parsePtr->string) - ? parsePtr->string : start - limit + 3, - (scanned < limit) ? scanned : limit - 3, start, - (scanned < limit) ? "" : "...", - insertMark ? mark : "", - (start + scanned + limit > parsePtr->end) - ? parsePtr->end - (start + scanned) : limit-3, - start + scanned, - (start + scanned + limit > parsePtr->end) ? "" : "..." - ); - if (post != NULL) { - Tcl_AppendToObj(msg, ";\n", -1); - Tcl_AppendObjToObj(msg, post); - Tcl_DecrRefCount(post); - } - Tcl_SetObjResult(interp, msg); - numBytes = parsePtr->end - parsePtr->string; - Tcl_AppendObjToErrorInfo(interp, Tcl_ObjPrintf( - "\n (parsing expression \"%.*s%s\")", - (numBytes < limit) ? numBytes : limit - 3, - parsePtr->string, (numBytes < limit) ? "" : "...")); + if (msg == NULL) { + msg = Tcl_GetObjResult(interp); + } + Tcl_AppendPrintfToObj(msg, "\nin expression \"%s%.*s%.*s%s%s%.*s%s\"", + ((start - limit) < parsePtr->string) ? "" : "...", + ((start - limit) < parsePtr->string) + ? (start - parsePtr->string) : limit - 3, + ((start - limit) < parsePtr->string) + ? parsePtr->string : start - limit + 3, + (scanned < limit) ? scanned : limit - 3, start, + (scanned < limit) ? "" : "...", insertMark ? mark : "", + (start + scanned + limit > parsePtr->end) + ? parsePtr->end - (start + scanned) : limit-3, + start + scanned, + (start + scanned + limit > parsePtr->end) ? "" : "..."); + if (post != NULL) { + Tcl_AppendToObj(msg, ";\n", -1); + Tcl_AppendObjToObj(msg, post); + Tcl_DecrRefCount(post); } + Tcl_SetObjResult(interp, msg); + numBytes = parsePtr->end - parsePtr->string; + Tcl_AppendObjToErrorInfo(interp, Tcl_ObjPrintf( + "\n (parsing expression \"%.*s%s\")", + (numBytes < limit) ? numBytes : limit - 3, + parsePtr->string, (numBytes < limit) ? "" : "...")); } return code; } #endif - -#ifndef PARSE_DIRECT_EXPR_TOKENS +#ifndef PARSE_DIRECT_EXPR_TOKENS /* *---------------------------------------------------------------------- * @@ -745,8 +776,8 @@ ParseExpr( * Number of bytes scanned. * * Side effects: - * The Tcl_Parse *parsePtr is filled with Tcl_Tokens representing - * the literal. + * The Tcl_Parse *parsePtr is filled with Tcl_Tokens representing the + * literal. * *---------------------------------------------------------------------- */ @@ -764,17 +795,21 @@ GenerateTokensForLiteral( Tcl_Token *destPtr; unsigned char lexeme; - /* Have to reparse to get pointers into source string */ + /* + * Have to reparse to get pointers into source string. + */ + scanned = TclParseAllWhiteSpace(start, numBytes); start +=scanned; scanned = ParseLexeme(start, numBytes-scanned, &lexeme, NULL); if ((lexeme != NUMBER) && (lexeme != BAREWORD)) { Tcl_Obj *literal; CONST char *bytes; + Tcl_ListObjIndex(NULL, litList, nextLiteral, &literal); bytes = Tcl_GetStringFromObj(literal, &scanned); start++; - if (memcmp((VOID *) bytes, (VOID *) start, (size_t) scanned) == 0) { + if (memcmp(bytes, start, (size_t) scanned) == 0) { closer = 1; } else { /* TODO */ @@ -809,8 +844,8 @@ GenerateTokensForLiteral( * Number of bytes scanned. * * Side effects: - * The Tcl_Parse *parsePtr is filled with Tcl_Tokens representing - * the literal. + * The Tcl_Parse *parsePtr is filled with Tcl_Tokens representing the + * literal. * *---------------------------------------------------------------------- */ @@ -828,8 +863,7 @@ CopyTokens( TclExpandTokenArray(parsePtr); } destPtr = parsePtr->tokenPtr + parsePtr->numTokens; - memcpy((VOID *) destPtr, (VOID *) sourcePtr, - (size_t) (toCopy * sizeof(Tcl_Token))); + memcpy(destPtr, sourcePtr, (size_t) toCopy * sizeof(Tcl_Token)); destPtr->type = TCL_TOKEN_SUB_EXPR; parsePtr->numTokens += toCopy; } else { @@ -841,8 +875,7 @@ CopyTokens( destPtr->type = TCL_TOKEN_SUB_EXPR; destPtr->numComponents++; destPtr++; - memcpy((VOID *) destPtr, (VOID *) sourcePtr, - (size_t) (toCopy * sizeof(Tcl_Token))); + memcpy(destPtr, sourcePtr, (size_t) toCopy * sizeof(Tcl_Token)); parsePtr->numTokens += toCopy + 1; } return toCopy; @@ -857,8 +890,8 @@ CopyTokens( * None. * * Side effects: - * The Tcl_Parse *parsePtr is filled with Tcl_Tokens representing - * the parsed expression. + * The Tcl_Parse *parsePtr is filled with Tcl_Tokens representing the + * parsed expression. * *---------------------------------------------------------------------- */ @@ -884,9 +917,13 @@ ConvertTreeToTokens( case UNARY: if (nodePtr->right > OT_NONE) { int right = nodePtr->right; + nodePtr->right = OT_NONE; if (nodePtr->lexeme != START) { - /* Find operator in string */ + /* + * Find operator in string. + */ + scanned = TclParseAllWhiteSpace(start, numBytes); start +=scanned; numBytes -= scanned; @@ -936,11 +973,17 @@ ConvertTreeToTokens( } } else { if (nodePtr->lexeme == START) { - /* We're done */ + /* + * We're done. + */ + return; } if (nodePtr->lexeme == OPEN_PAREN) { - /* Skip past matching close paren */ + /* + * Skip past matching close paren. + */ + scanned = TclParseAllWhiteSpace(start, numBytes); start +=scanned; numBytes -= scanned; @@ -960,6 +1003,7 @@ ConvertTreeToTokens( case BINARY: if (nodePtr->left > OT_NONE) { int left = nodePtr->left; + nodePtr->left = OT_NONE; scanned = TclParseAllWhiteSpace(start, numBytes); start +=scanned; @@ -995,6 +1039,7 @@ ConvertTreeToTokens( } } else if (nodePtr->right > OT_NONE) { int right = nodePtr->right; + nodePtr->right = OT_NONE; scanned = TclParseAllWhiteSpace(start, numBytes); start +=scanned; @@ -1038,7 +1083,7 @@ ConvertTreeToTokens( nodePtr->left = OT_NONE; destPtr = parsePtr->tokenPtr + tokenIdx; destPtr->size = start - destPtr->start; - destPtr->numComponents = parsePtr->numTokens - tokenIdx - 1; + destPtr->numComponents = parsePtr->numTokens-tokenIdx-1; } nodePtr = nodes + nodePtr->parent; } @@ -1047,7 +1092,6 @@ ConvertTreeToTokens( } } #endif - /* *---------------------------------------------------------------------- @@ -1060,11 +1104,11 @@ ConvertTreeToTokens( * caller. * * Results: - * If the string is successfully parsed as a valid Tcl expression, - * TCL_OK is returned, and data about the expression structure is - * written to *parsePtr. If the string cannot be parsed as a valid - * Tcl expression, TCL_ERROR is returned, and if interp is non-NULL, - * an error message is written to interp. + * If the string is successfully parsed as a valid Tcl expression, TCL_OK + * is returned, and data about the expression structure is written to + * *parsePtr. If the string cannot be parsed as a valid Tcl expression, + * TCL_ERROR is returned, and if interp is non-NULL, an error message is + * written to interp. * * Side effects: * If there is insufficient space in parsePtr to hold all the information @@ -1101,8 +1145,8 @@ Tcl_ParseExpr( TclParseInit(interp, start, numBytes, parsePtr); if (code == TCL_OK) { - ConvertTreeToTokens(interp, start, numBytes, opTree, - litList, parse.tokenPtr, parsePtr); + ConvertTreeToTokens(interp, start, numBytes, opTree, litList, + parse.tokenPtr, parsePtr); } else { /* TODO: copy over any error info to *parsePtr */ } @@ -1138,7 +1182,9 @@ Tcl_ParseExpr( TclParseInit(interp, start, numBytes, &scratch); TclParseInit(interp, start, numBytes, parsePtr); - /* Initialize the parse tree with the special "START" node */ + /* + * Initialize the parse tree with the special "START" node. + */ nodes->lexeme = START; nodes->left = -1; @@ -1153,9 +1199,10 @@ Tcl_ParseExpr( Tcl_Token *tokenPtr; /* - * Each pass through this loop adds one more ExprNode. - * Allocate space for one if required. + * Each pass through this loop adds one more ExprNode. Allocate space + * for one if required. */ + if (nodesUsed >= nodesAvailable) { int lastOrphanIdx = lastOrphanPtr - nodes; int size = nodesUsed * 2; @@ -1165,8 +1212,8 @@ Tcl_ParseExpr( nodes = NULL; } do { - newPtr = (ExprNode *) attemptckrealloc( (char *) nodes, - (unsigned int) (size * sizeof(ExprNode)) ); + newPtr = (ExprNode *) attemptckrealloc((char *) nodes, + (unsigned int) size * sizeof(ExprNode)); } while ((newPtr == NULL) && ((size -= (size - nodesUsed) / 2) > nodesUsed)); if (newPtr == NULL) { @@ -1177,8 +1224,8 @@ Tcl_ParseExpr( } nodesAvailable = size; if (nodes == NULL) { - memcpy((VOID *) newPtr, (VOID *) staticNodes, - (size_t) (nodesUsed * sizeof(ExprNode))); + memcpy(newPtr, staticNodes, + (size_t) nodesUsed * sizeof(ExprNode)); } nodes = newPtr; lastOrphanPtr = nodes + lastOrphanIdx; @@ -1186,7 +1233,9 @@ Tcl_ParseExpr( nodePtr = nodes + nodesUsed; lastNodePtr = nodePtr - 1; - /* Skip white space between lexemes */ + /* + * Skip white space between lexemes. + */ scanned = TclParseAllWhiteSpace(start, numBytes); start += scanned; @@ -1194,7 +1243,9 @@ Tcl_ParseExpr( scanned = ParseLexeme(start, numBytes, &(nodePtr->lexeme), NULL); - /* Use context to categorize the lexemes that are ambiguous */ + /* + * Use context to categorize the lexemes that are ambiguous. + */ if ((NODE_TYPE & nodePtr->lexeme) == 0) { switch (nodePtr->lexeme) { @@ -1248,7 +1299,9 @@ Tcl_ParseExpr( } } - /* Add node to parse tree based on category */ + /* + * Add node to parse tree based on category. + */ switch (NODE_TYPE & nodePtr->lexeme) { case LEAF: { @@ -1424,8 +1477,11 @@ Tcl_ParseExpr( if ((nodePtr->lexeme == CLOSE_PAREN) && (lastNodePtr->lexeme == OPEN_PAREN)) { if (lastNodePtr[-1].lexeme == FUNCTION) { - /* Normally, "()" is a syntax error, but as a special - * case accept it as an argument list for a function */ + /* + * Normally, "()" is a syntax error, but as a special case + * accept it as an argument list for a function. + */ + scanned = 0; nodePtr->lexeme = EMPTY; nodePtr->left = -1; @@ -1451,24 +1507,22 @@ Tcl_ParseExpr( if (lastNodePtr->lexeme == OPEN_PAREN) { msg = Tcl_NewStringObj("unbalanced open paren", -1); } else if (lastNodePtr->lexeme == COMMA) { - msg = Tcl_ObjPrintf( + msg = Tcl_ObjPrintf( "missing function argument at %s", mark); scanned = 0; insertMark = 1; } else if (lastNodePtr->lexeme == START) { msg = Tcl_NewStringObj("empty expression", -1); } - } else { - if (nodePtr->lexeme == CLOSE_PAREN) { - msg = Tcl_NewStringObj("unbalanced close paren", -1); - } else if ((nodePtr->lexeme == COMMA) - && (lastNodePtr->lexeme == OPEN_PAREN) - && (lastNodePtr[-1].lexeme == FUNCTION)) { - msg = Tcl_ObjPrintf( - "missing function argument at %s", mark); - scanned = 0; - insertMark = 1; - } + } else if (nodePtr->lexeme == CLOSE_PAREN) { + msg = Tcl_NewStringObj("unbalanced close paren", -1); + } else if ((nodePtr->lexeme == COMMA) + && (lastNodePtr->lexeme == OPEN_PAREN) + && (lastNodePtr[-1].lexeme == FUNCTION)) { + msg = Tcl_ObjPrintf( + "missing function argument at %s", mark); + scanned = 0; + insertMark = 1; } if (msg == NULL) { msg = Tcl_ObjPrintf("missing operand at %s", mark); @@ -1480,7 +1534,6 @@ Tcl_ParseExpr( } while (1) { - if (lastOrphanPtr->parent >= 0) { otherPtr = nodes + lastOrphanPtr->parent; } else if (lastOrphanPtr->left >= 0) { @@ -1496,8 +1549,11 @@ Tcl_ParseExpr( } if (prec[otherPtr->lexeme] == precedence) { - /* Special association rules for the ternary operators. */ - if ((otherPtr->lexeme == QUESTION) + /* + * Special association rules for the ternary operators. + */ + + if ((otherPtr->lexeme == QUESTION) && (lastOrphanPtr->lexeme != COLON)) { break; } @@ -1505,13 +1561,20 @@ Tcl_ParseExpr( && (nodePtr->lexeme == QUESTION)) { break; } - /* Right association rules for exponentiation. */ + + /* + * Right association rules for exponentiation. + */ + if (nodePtr->lexeme == EXPON) { break; } } - /* Some checks before linking */ + /* + * Some checks before linking. + */ + if ((otherPtr->lexeme == OPEN_PAREN) && (nodePtr->lexeme != CLOSE_PAREN)) { lastOrphanPtr = otherPtr; @@ -1537,19 +1600,28 @@ Tcl_ParseExpr( break; } - /* Link orphan as right operand of otherPtr */ + /* + * Link orphan as right operand of otherPtr. + */ + otherPtr->right = lastOrphanPtr - nodes; lastOrphanPtr->parent = otherPtr - nodes; lastOrphanPtr = otherPtr; if (otherPtr->lexeme == OPEN_PAREN) { - /* CLOSE_PAREN can only close one OPEN_PAREN */ + /* + * CLOSE_PAREN can only close one OPEN_PAREN. + */ + tokenPtr = scratch.tokenPtr + otherPtr->token; tokenPtr->size = start + scanned - tokenPtr->start; break; } if (otherPtr->lexeme == START) { - /* Don't backtrack beyond the start */ + /* + * Don't backtrack beyond the start. + */ + break; } } @@ -1563,7 +1635,11 @@ Tcl_ParseExpr( code = TCL_ERROR; continue; } - /* Create no node for a CLOSE_PAREN lexeme */ + + /* + * Create no node for a CLOSE_PAREN lexeme. + */ + break; } @@ -1583,7 +1659,10 @@ Tcl_ParseExpr( continue; } - /* Link orphan as left operand of new node */ + /* + * Link orphan as left operand of new node. + */ + nodePtr->right = -1; if (scratch.numTokens >= scratch.tokensAvailable) { @@ -1611,7 +1690,10 @@ Tcl_ParseExpr( } if (code == TCL_OK) { - /* Shift tokens from scratch space to caller space */ + /* + * Shift tokens from scratch space to caller space. + */ + GenerateTokens(nodes, &scratch, parsePtr); } else { if (parsePtr->errorType == TCL_PARSE_SUCCESS) { @@ -1626,7 +1708,8 @@ Tcl_ParseExpr( if (msg == NULL) { msg = Tcl_GetObjResult(interp); } - Tcl_AppendPrintfToObj(msg, "\nin expression \"%s%.*s%.*s%s%s%.*s%s\"", + Tcl_AppendPrintfToObj(msg, + "\nin expression \"%s%.*s%.*s%s%s%.*s%s\"", ((start - limit) < scratch.string) ? "" : "...", ((start - limit) < scratch.string) ? (start - scratch.string) : limit - 3, @@ -1638,8 +1721,7 @@ Tcl_ParseExpr( (start + scanned + limit > scratch.end) ? scratch.end - (start + scanned) : limit-3, start + scanned, - (start + scanned + limit > scratch.end) ? "" : "..." - ); + (start + scanned + limit > scratch.end) ? "" : "..."); if (post != NULL) { Tcl_AppendToObj(msg, ";\n", -1); Tcl_AppendObjToObj(msg, post); @@ -1661,19 +1743,18 @@ Tcl_ParseExpr( return code; #endif } - -#ifdef PARSE_DIRECT_EXPR_TOKENS +#ifdef PARSE_DIRECT_EXPR_TOKENS /* *---------------------------------------------------------------------- * * GenerateTokens -- * - * Routine that generates Tcl_Tokens that represent a Tcl expression - * and writes them to *parsePtr. The parse tree of the expression - * is in the array of ExprNodes, nodes. Some of the Tcl_Tokens are - * copied from scratch space at *scratchPtr, where the parsing pass - * that constructed the parse tree left them. + * Routine that generates Tcl_Tokens that represent a Tcl expression and + * writes them to *parsePtr. The parse tree of the expression is in the + * array of ExprNodes, nodes. Some of the Tcl_Tokens are copied from + * scratch space at *scratchPtr, where the parsing pass that constructed + * the parse tree left them. * *---------------------------------------------------------------------- */ @@ -1813,15 +1894,14 @@ GenerateTokens( } } #endif - /* *---------------------------------------------------------------------- * * ParseLexeme -- * - * Parse a single lexeme from the start of a string, scanning no - * more than numBytes bytes. + * Parse a single lexeme from the start of a string, scanning no more + * than numBytes bytes. * * Results: * Returns the number of bytes scanned to produce the lexeme. @@ -1838,8 +1918,8 @@ ParseLexeme( int numBytes, /* Number of bytes in string. */ unsigned char *lexemePtr, /* Write code of parsed lexeme to this * storage. */ - Tcl_Obj **literalPtr) /* Write corresponding literal value to - this storage, if non-NULL. */ + Tcl_Obj **literalPtr) /* Write corresponding literal value to this + storage, if non-NULL. */ { CONST char *end; int scanned; @@ -2073,10 +2153,10 @@ static int traceExprComp = 0; #endif /* TCL_COMPILE_DEBUG */ /* - * Definitions of numeric codes representing each expression operator. The - * order of these must match the entries in the operatorTable below. Also the + * Definitions of numeric codes representing each expression operator. The + * order of these must match the entries in the operatorTable below. Also the * codes for the relational operators (OP_LESS, OP_GREATER, OP_LE, OP_GE, - * OP_EQ, and OP_NE) must be consecutive and in that order. Note that OP_PLUS + * OP_EQ, and OP_NE) must be consecutive and in that order. Note that OP_PLUS * and OP_MINUS represent both unary and binary operators. */ @@ -2161,14 +2241,13 @@ static OperatorDesc operatorTable[] = { static Tcl_HashTable opHashTable; -#endif +#endif /* USE_EXPR_TOKENS */ /* * Declarations for local procedures to this file: */ #ifdef USE_EXPR_TOKENS - static void CompileCondExpr(Tcl_Interp *interp, Tcl_Token *exprTokenPtr, int *convertPtr, CompileEnv *envPtr); @@ -2181,8 +2260,7 @@ static void CompileMathFuncCall(Tcl_Interp *interp, static void CompileSubExpr(Tcl_Interp *interp, Tcl_Token *exprTokenPtr, int *convertPtr, CompileEnv *envPtr); -#endif - +#endif /* USE_EXPR_TOKENS */ static void CompileExprTree(Tcl_Interp *interp, OpNode *nodes, Tcl_Obj *const litObjv[], Tcl_Obj *funcList, Tcl_Token *tokenPtr, int *convertPtr, @@ -2247,10 +2325,13 @@ TclCompileExpr( Tcl_Obj **litObjv; /* TIP #280 : Track Lines within the expression */ - TclAdvanceLines (&envPtr->line, script, - script+TclParseAllWhiteSpace(script, numBytes)); + TclAdvanceLines(&envPtr->line, script, + script + TclParseAllWhiteSpace(script, numBytes)); + + /* + * Valid parse; compile the tree. + */ - /* Valid parse; compile the tree */ Tcl_ListObjGetElements(NULL, litList, &litObjc, &litObjv); CompileExprTree(interp, opTree, litObjv, funcList, parse.tokenPtr, &needsNumConversion, envPtr); @@ -2261,6 +2342,7 @@ TclCompileExpr( * operands if at all possible as first integers, else * floating-point numbers. */ + TclEmitOpcode(INST_TRY_CVT_TO_NUMERIC, envPtr); } } @@ -2286,9 +2368,11 @@ TclCompileExpr( Tcl_MutexLock(&opMutex); if (!opTableInitialized) { int i; + Tcl_InitHashTable(&opHashTable, TCL_STRING_KEYS); for (i = 0; operatorTable[i].name != NULL; i++) { int new; + Tcl_HashEntry *hPtr = Tcl_CreateHashEntry(&opHashTable, operatorTable[i].name, &new); if (new) { @@ -2315,7 +2399,7 @@ TclCompileExpr( if (needsNumConversion) { /* - * Attempt to convert the primary's object to an int or double. This + * Attempt to convert the primary's object to an int or double. This * is done in order to support Tcl's policy of interpreting operands * if at all possible as first integers, else floating-point numbers. */ @@ -2327,13 +2411,12 @@ TclCompileExpr( return TCL_OK; #endif } - /* *---------------------------------------------------------------------- * * CompileExprTree -- - * + * [???] * * Results: * None. @@ -2354,9 +2437,9 @@ typedef struct JumpList { static void CompileExprTree( - Tcl_Interp *interp, + Tcl_Interp *interp, OpNode *nodes, - Tcl_Obj * const litObjv[], + Tcl_Obj *const litObjv[], Tcl_Obj *funcList, Tcl_Token *tokenPtr, int *convertPtr, @@ -2387,12 +2470,14 @@ CompileExprTree( case UNARY: if (nodePtr->right > OT_NONE) { int right = nodePtr->right; + nodePtr->right = OT_NONE; if (nodePtr->lexeme == FUNCTION) { Tcl_DString cmdName; - Tcl_Obj *funcName; + Tcl_Obj *funcName; CONST char *p; int length; + Tcl_DStringInit(&cmdName); Tcl_DStringAppend(&cmdName, "tcl::mathfunc::", -1); Tcl_ListObjIndex(NULL, funcList, nextFunc++, &funcName); @@ -2413,7 +2498,8 @@ CompileExprTree( break; case OT_TOKENS: if (tokenPtr->type != TCL_TOKEN_WORD) { - Tcl_Panic("unexpected token type %d\n", tokenPtr->type); + Tcl_Panic("unexpected token type %d\n", + tokenPtr->type); } TclCompileTokens(interp, tokenPtr+1, tokenPtr->numComponents, envPtr); @@ -2431,7 +2517,7 @@ CompileExprTree( /* do nothing */ } else if (nodePtr->lexeme == FUNCTION) { int numWords = (nodePtr[1].left - OT_NONE) + 1; - if ( numWords < 255) { + if (numWords < 255) { TclEmitInstInt1(INST_INVOKE_STK1, numWords, envPtr); } else { TclEmitInstInt4(INST_INVOKE_STK4, numWords, envPtr); @@ -2454,19 +2540,18 @@ CompileExprTree( TclStackAlloc(interp, sizeof(JumpList)); newJump->next = jumpPtr; jumpPtr = newJump; - newJump = (JumpList *) + newJump = (JumpList *) TclStackAlloc(interp, sizeof(JumpList)); newJump->next = jumpPtr; jumpPtr = newJump; jumpPtr->depth = envPtr->currStackDepth; *convertPtr = 1; - } else if ((nodePtr->lexeme == AND) - || (nodePtr->lexeme == OR)) { + } else if (nodePtr->lexeme == AND || nodePtr->lexeme == OR) { JumpList *newJump = (JumpList *) TclStackAlloc(interp, sizeof(JumpList)); newJump->next = jumpPtr; jumpPtr = newJump; - newJump = (JumpList *) + newJump = (JumpList *) TclStackAlloc(interp, sizeof(JumpList)); newJump->next = jumpPtr; jumpPtr = newJump; @@ -2478,12 +2563,13 @@ CompileExprTree( } switch (left) { case OT_LITERAL: - TclEmitPush( TclAddLiteralObj( - envPtr, *litObjv++, NULL), envPtr); + TclEmitPush(TclAddLiteralObj(envPtr, *litObjv++, NULL), + envPtr); break; case OT_TOKENS: if (tokenPtr->type != TCL_TOKEN_WORD) { - Tcl_Panic("unexpected token type %d\n", tokenPtr->type); + Tcl_Panic("unexpected token type %d\n", + tokenPtr->type); } TclCompileTokens(interp, tokenPtr+1, tokenPtr->numComponents, envPtr); @@ -2494,6 +2580,7 @@ CompileExprTree( } } else if (nodePtr->right > OT_NONE) { int right = nodePtr->right; + nodePtr->right = OT_NONE; if (nodePtr->lexeme == QUESTION) { TclEmitForwardJump(envPtr, TCL_FALSE_JUMP, @@ -2514,12 +2601,13 @@ CompileExprTree( } switch (right) { case OT_LITERAL: - TclEmitPush( TclAddLiteralObj( - envPtr, *litObjv++, NULL), envPtr); + TclEmitPush(TclAddLiteralObj(envPtr, *litObjv++, NULL), + envPtr); break; case OT_TOKENS: if (tokenPtr->type != TCL_TOKEN_WORD) { - Tcl_Panic("unexpected token type %d\n", tokenPtr->type); + Tcl_Panic("unexpected token type %d\n", + tokenPtr->type); } TclCompileTokens(interp, tokenPtr+1, tokenPtr->numComponents, envPtr); @@ -2529,8 +2617,7 @@ CompileExprTree( nodePtr = nodes + right; } } else { - if ((nodePtr->lexeme == COMMA) - || (nodePtr->lexeme == QUESTION)) { + if (nodePtr->lexeme == COMMA || nodePtr->lexeme == QUESTION) { /* do nothing */ } else if (nodePtr->lexeme == COLON) { if (TclFixupForwardJump(envPtr, &(jumpPtr->next->jump), @@ -2595,9 +2682,12 @@ OpCmd( int code, tmp=1; Tcl_Obj *byteCodeObj = Tcl_NewObj(); - /* Note we are compiling an expression with literal arguments. - * This means there can be no [info frame] calls when we execute - * the resulting bytecode, so there's no need to tend to TIP 280 issues */ + /* + * Note we are compiling an expression with literal arguments. This means + * there can be no [info frame] calls when we execute the resulting + * bytecode, so there's no need to tend to TIP 280 issues. + */ + TclInitCompileEnv(interp, &compEnv, NULL, 0, NULL, 0); CompileExprTree(interp, nodes, litObjv, NULL, NULL, &tmp, &compEnv); TclEmitOpcode(INST_DONE, &compEnv); @@ -2830,9 +2920,8 @@ TclFinalizeCompilation(void) Tcl_MutexUnlock(&opMutex); #endif } - -#ifdef USE_EXPR_TOKENS +#ifdef USE_EXPR_TOKENS /* *---------------------------------------------------------------------- * @@ -2861,7 +2950,10 @@ CompileSubExpr( * not needed */ CompileEnv *envPtr) /* Holds resulting instructions. */ { - /* Switch on the type of the first token after the subexpression token. */ + /* + * Switch on the type of the first token after the subexpression token. + */ + Tcl_Token *tokenPtr = exprTokenPtr+1; TRACE(exprTokenPtr->start, exprTokenPtr->size, tokenPtr->start, tokenPtr->size); @@ -2896,9 +2988,10 @@ CompileSubExpr( case TCL_TOKEN_OPERATOR: { /* - * Look up the operator. If the operator isn't found, treat it as a + * Look up the operator. If the operator isn't found, treat it as a * math function. */ + OperatorDesc *opDescPtr; Tcl_HashEntry *hPtr; CONST char *operator; @@ -3067,7 +3160,7 @@ CompileLandOrLorExpr( TclEmitForwardJump(envPtr, TCL_UNCONDITIONAL_JUMP, &endFixup); /* - * Fixup the short-circuit jumps and push the shortCircuit value. Note + * Fixup the short-circuit jumps and push the shortCircuit value. Note * that shortCircuitFixup2 is always a short jump. */ @@ -3243,12 +3336,15 @@ CompileMathFuncCall( afterSubexprPtr = exprTokenPtr + (exprTokenPtr->numComponents + 1); while (tokenPtr != afterSubexprPtr) { int convert = 0; + ++argCount; CompileSubExpr(interp, tokenPtr, &convert, envPtr); tokenPtr += (tokenPtr->numComponents + 1); } - /* Invoke the function */ + /* + * Invoke the function. + */ if (argCount < 255) { TclEmitInstInt1(INST_INVOKE_STK1, argCount, envPtr); @@ -3257,7 +3353,6 @@ CompileMathFuncCall( } } #endif - /* * Local Variables: |