summaryrefslogtreecommitdiffstats
path: root/generic/tclCompCmds.c
diff options
context:
space:
mode:
Diffstat (limited to 'generic/tclCompCmds.c')
-rw-r--r--generic/tclCompCmds.c422
1 files changed, 227 insertions, 195 deletions
diff --git a/generic/tclCompCmds.c b/generic/tclCompCmds.c
index f5c553a..79e1640 100644
--- a/generic/tclCompCmds.c
+++ b/generic/tclCompCmds.c
@@ -12,7 +12,7 @@
* See the file "license.terms" for information on usage and redistribution of
* this file, and for a DISCLAIMER OF ALL WARRANTIES.
*
- * RCS: @(#) $Id: tclCompCmds.c,v 1.98 2007/01/09 11:32:33 dkf Exp $
+ * RCS: @(#) $Id: tclCompCmds.c,v 1.99 2007/02/27 21:28:45 dkf Exp $
*/
#include "tclInt.h"
@@ -36,17 +36,18 @@
(envPtr)); \
}
-/* TIP #280 : Remember the per-word line information of the current
- * command. An index is used instead of a pointer as recursive compilation may
- * reallocate, i.e. move, the array. This is also the reason to save the nuloc
- * now, it may change during the course of the function.
+/*
+ * TIP #280: Remember the per-word line information of the current command. An
+ * index is used instead of a pointer as recursive compilation may reallocate,
+ * i.e. move, the array. This is also the reason to save the nuloc now, it may
+ * change during the course of the function.
*
* Macro to encapsulate the variable definition and setup.
*/
#define DefineLineInformation \
- ExtCmdLoc* mapPtr = envPtr->extCmdMapPtr; \
- int eclIndex = mapPtr->nuloc - 1
+ ExtCmdLoc *mapPtr = envPtr->extCmdMapPtr; \
+ int eclIndex = mapPtr->nuloc - 1
/*
* Convenience macro for use when compiling bodies of commands. The ANSI C
@@ -207,11 +208,13 @@ TclCompileAppendCmd(
/*
* append varName == set varName
*/
+
return TclCompileSetCmd(interp, parsePtr, envPtr);
} else if (numWords > 3) {
/*
- * APPEND instructions currently only handle one value
+ * APPEND instructions currently only handle one value.
*/
+
return TCL_ERROR;
}
@@ -342,6 +345,7 @@ TclCompileCatchCmd(
* If syntax does not match what we expect for [catch], do not compile.
* Let runtime checks determine if syntax has changed.
*/
+
if ((parsePtr->numWords < 2) || (parsePtr->numWords > 4)) {
return TCL_ERROR;
}
@@ -365,18 +369,19 @@ TclCompileCatchCmd(
if (parsePtr->numWords >= 3) {
resultNameTokenPtr = TokenAfter(cmdTokenPtr);
/* DGP */
- if (resultNameTokenPtr->type == TCL_TOKEN_SIMPLE_WORD) {
- name = resultNameTokenPtr[1].start;
- nameChars = resultNameTokenPtr[1].size;
- if (!TclIsLocalScalar(name, nameChars)) {
- return TCL_ERROR;
- }
- resultIndex = TclFindCompiledLocal(resultNameTokenPtr[1].start,
- resultNameTokenPtr[1].size, /*create*/ 1, VAR_SCALAR,
- envPtr->procPtr);
- } else {
+ if (resultNameTokenPtr->type != TCL_TOKEN_SIMPLE_WORD) {
return TCL_ERROR;
}
+
+ name = resultNameTokenPtr[1].start;
+ nameChars = resultNameTokenPtr[1].size;
+ if (!TclIsLocalScalar(name, nameChars)) {
+ return TCL_ERROR;
+ }
+ resultIndex = TclFindCompiledLocal(resultNameTokenPtr[1].start,
+ resultNameTokenPtr[1].size, /*create*/ 1, VAR_SCALAR,
+ envPtr->procPtr);
+
/* DKF */
if (parsePtr->numWords == 4) {
optsNameTokenPtr = TokenAfter(resultNameTokenPtr);
@@ -687,6 +692,7 @@ TclCompileDictCmd(
/*
* Only compile this because we need INST_DICT_GET anyway.
*/
+
if (numWords < 2) {
return TCL_ERROR;
}
@@ -910,7 +916,7 @@ TclCompileDictCmd(
procPtr);
Tcl_DStringInit(&localVarsLiteral);
- keyTokenPtrs = (Tcl_Token **) ckalloc(sizeof(Tcl_Token*) * numVars);
+ keyTokenPtrs = (Tcl_Token **) ckalloc(sizeof(Tcl_Token *) * numVars);
tokenPtr = TokenAfter(dictVarTokenPtr);
for (i=0 ; i<numVars ; i++) {
keyTokenPtrs[i] = tokenPtr;
@@ -970,9 +976,11 @@ TclCompileDictCmd(
TclEmitInstInt4( INST_LOAD_SCALAR4, keyTmpIndex, envPtr);
PushLiteral(envPtr, Tcl_DStringValue(&localVarsLiteral),
Tcl_DStringLength(&localVarsLiteral));
+
/*
* Any literal would do, but this one is handy...
*/
+
TclEmitInstInt4( INST_STORE_SCALAR4, keyTmpIndex, envPtr);
TclEmitInstInt4( INST_DICT_UPDATE_END, dictIndex, envPtr);
@@ -990,6 +998,7 @@ TclCompileDictCmd(
* Arbirary safe limit; anyone exceeding it should stop worrying about
* speed quite so much. ;-)
*/
+
if (numWords < 3 || numWords > 100 || procPtr == NULL) {
return TCL_ERROR;
}
@@ -1044,6 +1053,7 @@ TclCompileDictCmd(
/*
* Something we do not know how to compile.
*/
+
return TCL_ERROR;
}
@@ -1079,7 +1089,7 @@ TclCompileExprCmd(
}
/*
- * TIP #280 : Use the per-word line information of the current command.
+ * TIP #280: Use the per-word line information of the current command.
*/
envPtr->line = envPtr->extCmdMapPtr->loc[
@@ -1271,7 +1281,7 @@ TclCompileForCmd(
* Instructions are added to envPtr to execute the "foreach" command at
* runtime.
*
-n*----------------------------------------------------------------------
+ *----------------------------------------------------------------------
*/
int
@@ -1300,8 +1310,8 @@ TclCompileForeachCmd(
/*
* We parse the variable list argument words and create two arrays:
- * varcList[i] is number of variables in i-th var list
- * varvList[i] points to array of var names in i-th var list
+ * varcList[i] is number of variables in i-th var list.
+ * varvList[i] points to array of var names in i-th var list.
*/
#define STATIC_VAR_LIST_SIZE 5
@@ -1326,8 +1336,9 @@ TclCompileForeachCmd(
/*
* Bail out if the body requires substitutions in order to insure correct
- * behaviour [Bug 219166]
+ * behaviour. [Bug 219166]
*/
+
for (i = 0, tokenPtr = parsePtr->tokenPtr; i < numWords-1; i++) {
tokenPtr = TokenAfter(tokenPtr);
}
@@ -1675,6 +1686,7 @@ FreeForeachInfo(
*
*----------------------------------------------------------------------
*/
+
int
TclCompileIfCmd(
Tcl_Interp *interp, /* Used for error reporting. */
@@ -1700,7 +1712,7 @@ TclCompileIfCmd(
* to this value at the start of each test. */
int realCond = 1; /* Set to 0 for static conditions:
* "if 0 {..}" */
- int boolVal; /* Value of static condition */
+ int boolVal; /* Value of static condition. */
int compileScripts = 1;
DefineLineInformation; /* TIP #280 */
@@ -1772,8 +1784,9 @@ TclCompileIfCmd(
TclDecrRefCount(boolObj);
if (code == TCL_OK) {
/*
- * A static condition
+ * A static condition.
*/
+
realCond = 0;
if (!boolVal) {
compileScripts = 0;
@@ -1938,7 +1951,7 @@ TclCompileIfCmd(
*/
for (j = jumpEndFixupArray.next; j > 0; j--) {
- jumpIndex = (j - 1); /* i.e. process the closest jump first */
+ jumpIndex = (j - 1); /* i.e. process the closest jump first. */
if (TclFixupForwardJumpToHere(envPtr,
jumpEndFixupArray.fixup+jumpIndex, 127)) {
/*
@@ -1949,6 +1962,7 @@ TclCompileIfCmd(
unsigned char *ifFalsePc = envPtr->codeStart
+ jumpFalseFixupArray.fixup[jumpIndex].codeOffset;
unsigned char opCode = *ifFalsePc;
+
if (opCode == INST_JUMP_FALSE1) {
jumpFalseDist = TclGetInt1AtPtr(ifFalsePc + 1);
jumpFalseDist += 3;
@@ -2040,7 +2054,7 @@ TclCompileIncrCmd(
envPtr->line = mapPtr->loc[eclIndex].line[2];
CompileTokens(envPtr, incrTokenPtr, interp);
}
- } else { /* No incr amount given so use 1 */
+ } else { /* No incr amount given so use 1. */
haveImmValue = 1;
}
@@ -2080,7 +2094,7 @@ TclCompileIncrCmd(
}
}
}
- } else { /* Non-simple variable name */
+ } else { /* Non-simple variable name. */
if (haveImmValue) {
TclEmitInstInt1(INST_INCR_STK_IMM, immValue, envPtr);
} else {
@@ -2123,6 +2137,7 @@ TclCompileLappendCmd(
/*
* If we're not in a procedure, don't compile.
*/
+
if (envPtr->procPtr == NULL) {
return TCL_ERROR;
}
@@ -2133,8 +2148,9 @@ TclCompileLappendCmd(
}
if (numWords != 3) {
/*
- * LAPPEND instructions currently only handle one value appends
+ * LAPPEND instructions currently only handle one value appends.
*/
+
return TCL_ERROR;
}
@@ -2170,6 +2186,7 @@ TclCompileLappendCmd(
* The *_STK opcodes should be refactored to make better use of existing
* LOAD/STORE instructions.
*/
+
if (simpleVarName) {
if (isScalar) {
if (localIndex < 0) {
@@ -2225,9 +2242,11 @@ TclCompileLassignCmd(
DefineLineInformation; /* TIP #280 */
numWords = parsePtr->numWords;
+
/*
- * Check for command syntax error, but we'll punt that to runtime
+ * Check for command syntax error, but we'll punt that to runtime.
*/
+
if (numWords < 3) {
return TCL_ERROR;
}
@@ -2235,35 +2254,38 @@ TclCompileLassignCmd(
/*
* Generate code to push list being taken apart by [lassign].
*/
+
tokenPtr = TokenAfter(parsePtr->tokenPtr);
CompileWord(envPtr, tokenPtr, interp, 1);
/*
- * Generate code to assign values from the list to variables
+ * Generate code to assign values from the list to variables.
*/
+
for (idx=0 ; idx<numWords-2 ; idx++) {
tokenPtr = TokenAfter(tokenPtr);
/*
- * Generate the next variable name
+ * Generate the next variable name.
*/
- PushVarName(interp, tokenPtr, envPtr, TCL_CREATE_VAR,
- &localIndex, &simpleVarName, &isScalar,
- mapPtr->loc[eclIndex].line[idx+2]);
+
+ PushVarName(interp, tokenPtr, envPtr, TCL_CREATE_VAR, &localIndex,
+ &simpleVarName, &isScalar, mapPtr->loc[eclIndex].line[idx+2]);
/*
* Emit instructions to get the idx'th item out of the list value on
* the stack and assign it to the variable.
*/
+
if (simpleVarName) {
if (isScalar) {
if (localIndex >= 0) {
TclEmitOpcode(INST_DUP, envPtr);
TclEmitInstInt4(INST_LIST_INDEX_IMM, idx, envPtr);
if (localIndex <= 255) {
- TclEmitInstInt1(INST_STORE_SCALAR1, localIndex, envPtr);
+ TclEmitInstInt1(INST_STORE_SCALAR1,localIndex,envPtr);
} else {
- TclEmitInstInt4(INST_STORE_SCALAR4, localIndex, envPtr);
+ TclEmitInstInt4(INST_STORE_SCALAR4,localIndex,envPtr);
}
} else {
TclEmitInstInt4(INST_OVER, 1, envPtr);
@@ -2296,6 +2318,7 @@ TclCompileLassignCmd(
/*
* Generate code to leave the rest of the list on the stack.
*/
+
TclEmitInstInt4(INST_LIST_RANGE_IMM, idx, envPtr);
TclEmitInt4(-2, envPtr); /* -2 == "end" */
@@ -2332,7 +2355,7 @@ TclCompileLindexCmd(
DefineLineInformation; /* TIP #280 */
/*
- * Quit if too few args
+ * Quit if too few args.
*/
if (numWords <= 1) {
@@ -2427,6 +2450,7 @@ TclCompileListCmd(
/*
* If we're not in a procedure, don't compile.
*/
+
if (envPtr->procPtr == NULL) {
return TCL_ERROR;
}
@@ -2441,6 +2465,7 @@ TclCompileListCmd(
/*
* Push the all values onto the stack.
*/
+
Tcl_Token *valueTokenPtr;
int i, numWords;
@@ -2537,18 +2562,18 @@ TclCompileLlengthCmd(
int
TclCompileLsetCmd(
- Tcl_Interp *interp, /* Tcl interpreter for error reporting */
+ Tcl_Interp *interp, /* Tcl interpreter for error reporting. */
Tcl_Parse *parsePtr, /* Points to a parse structure for the
- * command */
- CompileEnv *envPtr) /* Holds the resulting instructions */
+ * command. */
+ CompileEnv *envPtr) /* Holds the resulting instructions. */
{
int tempDepth; /* Depth used for emitting one part of the
* code burst. */
Tcl_Token *varTokenPtr; /* Pointer to the Tcl_Token representing the
- * parse of the variable name */
- int localIndex; /* Index of var in local var table */
- int simpleVarName; /* Flag == 1 if var name is simple */
- int isScalar; /* Flag == 1 if scalar, 0 if array */
+ * parse of the variable name. */
+ int localIndex; /* Index of var in local var table. */
+ int simpleVarName; /* Flag == 1 if var name is simple. */
+ int isScalar; /* Flag == 1 if scalar, 0 if array. */
int i;
DefineLineInformation; /* TIP #280 */
@@ -2600,7 +2625,7 @@ TclCompileLsetCmd(
}
/*
- * Duplicate an array index if one's been pushed
+ * Duplicate an array index if one's been pushed.
*/
if (simpleVarName && !isScalar) {
@@ -2637,7 +2662,7 @@ TclCompileLsetCmd(
}
/*
- * Emit the correct variety of 'lset' instruction
+ * Emit the correct variety of 'lset' instruction.
*/
if (parsePtr->numWords == 4) {
@@ -2647,7 +2672,7 @@ TclCompileLsetCmd(
}
/*
- * Emit code to put the value back in the variable
+ * Emit code to put the value back in the variable.
*/
if (!simpleVarName) {
@@ -2693,13 +2718,13 @@ TclCompileLsetCmd(
int
TclCompileRegexpCmd(
- Tcl_Interp *interp, /* Tcl interpreter for error reporting */
+ Tcl_Interp *interp, /* Tcl interpreter for error reporting. */
Tcl_Parse *parsePtr, /* Points to a parse structure for the
- * command */
- CompileEnv *envPtr) /* Holds the resulting instructions */
+ * command. */
+ CompileEnv *envPtr) /* Holds the resulting instructions. */
{
Tcl_Token *varTokenPtr; /* Pointer to the Tcl_Token representing the
- * parse of the RE or string */
+ * parse of the RE or string. */
int i, len, nocase, anchorLeft, anchorRight, start;
char *str;
DefineLineInformation; /* TIP #280 */
@@ -2728,7 +2753,7 @@ TclCompileRegexpCmd(
varTokenPtr = TokenAfter(varTokenPtr);
if (varTokenPtr->type != TCL_TOKEN_SIMPLE_WORD) {
/*
- * Not a simple string - punt to runtime.
+ * Not a simple string, so punt to runtime.
*/
return TCL_ERROR;
@@ -2861,7 +2886,7 @@ TclCompileRegexpCmd(
ckfree((char *) str);
/*
- * Push the string arg
+ * Push the string arg.
*/
varTokenPtr = TokenAfter(varTokenPtr);
@@ -2905,14 +2930,13 @@ TclCompileReturnCmd(
* General syntax: [return ?-option value ...? ?result?]
* An even number of words means an explicit result argument is present.
*/
- int level, code, status = TCL_OK;
+ int level, code, objc, status = TCL_OK;
int numWords = parsePtr->numWords;
int explicitResult = (0 == (numWords % 2));
int numOptionWords = numWords - 1 - explicitResult;
Tcl_Obj *returnOpts;
Tcl_Token *wordTokenPtr = TokenAfter(parsePtr->tokenPtr);
#define NUM_STATIC_OBJS 20
- int objc;
Tcl_Obj *staticObjArray[NUM_STATIC_OBJS], **objv;
DefineLineInformation; /* TIP #280 */
@@ -2938,7 +2962,7 @@ TclCompileReturnCmd(
}
/*
- * Allocate some working space if needed
+ * Allocate some working space if needed.
*/
if (numOptionWords > NUM_STATIC_OBJS) {
@@ -2978,13 +3002,14 @@ TclCompileReturnCmd(
* and report back to the compiler that this must be interpreted at
* runtime.
*/
+
Tcl_ResetResult(interp);
return TCL_ERROR;
}
/*
* All options are known at compile time, so we're going to bytecompile.
- * Emit instructions to push the result on the stack
+ * Emit instructions to push the result on the stack.
*/
if (explicitResult) {
@@ -2993,6 +3018,7 @@ TclCompileReturnCmd(
/*
* No explict result argument, so default result is empty string.
*/
+
PushLiteral(envPtr, "", 0);
}
@@ -3024,6 +3050,7 @@ TclCompileReturnCmd(
* ... and there is no enclosing catch. Issue the maximally
* efficient exit instruction.
*/
+
Tcl_DecrRefCount(returnOpts);
TclEmitOpcode(INST_DONE, envPtr);
return TCL_OK;
@@ -3347,6 +3374,7 @@ TclCompileStringCmd(
/*
* All other cases: compile out of line.
*/
+
return TCL_ERROR;
}
@@ -3383,8 +3411,8 @@ TclCompileSwitchCmd(
* created by Tcl_ParseCommand. */
CompileEnv *envPtr) /* Holds resulting instructions. */
{
- Tcl_Token *tokenPtr; /* Pointer to tokens in command */
- int numWords; /* Number of words in command */
+ Tcl_Token *tokenPtr; /* Pointer to tokens in command. */
+ int numWords; /* Number of words in command. */
Tcl_Token *valueTokenPtr; /* Token for the value to switch on. */
enum {Switch_Exact, Switch_Glob} mode;
@@ -3392,7 +3420,8 @@ TclCompileSwitchCmd(
Tcl_Token *bodyTokenArray; /* Array of real pattern list items. */
Tcl_Token **bodyToken; /* Array of pointers to pattern list items. */
- int *bodyLines; /* Array of line numbers for body list items */
+ int *bodyLines; /* Array of line numbers for body list
+ * items. */
int foundDefault; /* Flag to indicate whether a "default" clause
* is present. */
@@ -3499,6 +3528,7 @@ TclCompileSwitchCmd(
/*
* Can't compile this case; no opcode for case-insensitive equality!
*/
+
return TCL_ERROR;
}
@@ -3509,7 +3539,7 @@ TclCompileSwitchCmd(
*/
valueTokenPtr = tokenPtr;
- /* valueIndex see previous loop */
+ /* For valueIndex, see previous loop. */
tokenPtr = TokenAfter(tokenPtr);
numWords--;
@@ -3523,18 +3553,12 @@ TclCompileSwitchCmd(
if (numWords == 1) {
Tcl_DString bodyList;
- const char **argv = NULL;
+ const char **argv = NULL, *tokenStartPtr, *p;
+ int bline; /* TIP #280: line of the pattern/action list,
+ * and start of list for when tracking the
+ * location. This list comes immediately after
+ * the value we switch on. */
int isTokenBraced;
- const char *tokenStartPtr;
-
- /*
- * TIP #280: line of the pattern/action list, and start of list for
- * when tracking the location. This list comes immediately after the
- * value we switch on.
- */
-
- int bline = mapPtr->loc[eclIndex].line[valueIndex+1];
- const char* p;
/*
* Test that we've got a suitable body list as a simple (i.e. braced)
@@ -3575,6 +3599,7 @@ TclCompileSwitchCmd(
* Locate the start of the arms within the overall word.
*/
+ bline = mapPtr->loc[eclIndex].line[valueIndex+1];
p = tokenStartPtr = tokenPtr[1].start;
while (isspace(UCHAR(*tokenStartPtr))) {
tokenStartPtr++;
@@ -3587,7 +3612,7 @@ TclCompileSwitchCmd(
}
/*
- * TIP #280. Count lines within the literal list.
+ * TIP #280: Count lines within the literal list.
*/
for (i=0 ; i<numWords ; i++) {
@@ -3616,7 +3641,7 @@ TclCompileSwitchCmd(
}
/*
- * TIP #280 Now determine the line the list element starts on
+ * TIP #280: Now determine the line the list element starts on
* (there is no need to do it earlier, due to the possibility of
* aborting, see above).
*/
@@ -3687,7 +3712,7 @@ TclCompileSwitchCmd(
bodyToken[i] = tokenPtr+1;
/*
- * TIP#280: Copy line information from regular cmd info.
+ * TIP #280: Copy line information from regular cmd info.
*/
bodyLines[i] = mapPtr->loc[eclIndex].line[valueIndex+1+i];
@@ -3836,7 +3861,7 @@ TclCompileSwitchCmd(
* Compile the body of the arm.
*/
- envPtr->line = bodyLines[i+1]; /* TIP#280 */
+ envPtr->line = bodyLines[i+1]; /* TIP #280 */
TclCompileCmdWord(interp, bodyToken[i+1], 1, envPtr);
/*
@@ -3987,7 +4012,7 @@ TclCompileSwitchCmd(
TclEmitOpcode(INST_POP, envPtr);
envPtr->currStackDepth = savedStackDepth + 1;
- envPtr->line = bodyLines[i+1]; /* #280 */
+ envPtr->line = bodyLines[i+1]; /* TIP #280 */
TclCompileCmdWord(interp, bodyToken[i+1], 1, envPtr);
if (!foundDefault) {
@@ -4147,6 +4172,7 @@ TclCompileVariableCmd(
/*
* Skip non-literals.
*/
+
if (varTokenPtr->type != TCL_TOKEN_SIMPLE_WORD) {
continue;
}
@@ -4157,6 +4183,7 @@ TclCompileVariableCmd(
/*
* Skip if it looks like it might be an array or an empty string.
*/
+
if ((*tail == ')') || (tail < varName)) {
continue;
}
@@ -4280,7 +4307,7 @@ TclCompileWhileCmd(
if (loopMayEnd) {
TclEmitForwardJump(envPtr, TCL_UNCONDITIONAL_JUMP, &jumpEvalCondFixup);
- testCodeOffset = 0; /* Avoid compiler warning */
+ testCodeOffset = 0; /* Avoid compiler warning. */
} else {
testCodeOffset = CurrentOffset(envPtr);
}
@@ -4372,11 +4399,11 @@ PushVarName(
Tcl_Interp *interp, /* Used for error reporting. */
Tcl_Token *varTokenPtr, /* Points to a variable token. */
CompileEnv *envPtr, /* Holds resulting instructions. */
- int flags, /* TCL_CREATE_VAR or TCL_NO_LARGE_INDEX */
- int *localIndexPtr, /* Must not be NULL */
- int *simpleVarNamePtr, /* Must not be NULL */
- int *isScalarPtr, /* Must not be NULL */
- int line) /* line the token starts on */
+ int flags, /* TCL_CREATE_VAR or TCL_NO_LARGE_INDEX. */
+ int *localIndexPtr, /* Must not be NULL. */
+ int *simpleVarNamePtr, /* Must not be NULL. */
+ int *isScalarPtr, /* Must not be NULL. */
+ int line) /* Line the token starts on. */
{
register const char *p;
const char *name, *elName;
@@ -4415,6 +4442,7 @@ PushVarName(
* A simple variable name. Divide it up into "name" and "elName"
* strings. If it is not a local variable, look it up at runtime.
*/
+
simpleVarName = 1;
name = varTokenPtr[1].start;
@@ -4454,7 +4482,7 @@ PushVarName(
&& (varTokenPtr[n].start[varTokenPtr[n].size - 1] == ')')) {
/*
- * Check for parentheses inside first token
+ * Check for parentheses inside first token.
*/
simpleVarName = 0;
@@ -4599,8 +4627,8 @@ PushVarName(
* evaluation to runtime.
*
* Side effects:
- * Instructions are added to envPtr to execute the compiled
- * command at runtime.
+ * Instructions are added to envPtr to execute the compiled command at
+ * runtime.
*
*----------------------------------------------------------------------
*/
@@ -4629,19 +4657,19 @@ CompileUnaryOpCmd(
*
* CompileAssociativeBinaryOpCmd --
*
- * Utility routine to compile the binary operator commands that
- * accept an arbitrary number of arguments, and that are associative
- * operations. Because of the associativity, we may combine operations
- * from right to left, saving us any effort of re-ordering the arguments
- * on the stack after substitutions are completed.
+ * Utility routine to compile the binary operator commands that accept an
+ * arbitrary number of arguments, and that are associative operations.
+ * Because of the associativity, we may combine operations from right to
+ * left, saving us any effort of re-ordering the arguments on the stack
+ * after substitutions are completed.
*
* Results:
* Returns TCL_OK for a successful compile. Returns TCL_ERROR to defer
* evaluation to runtime.
*
* Side effects:
- * Instructions are added to envPtr to execute the compiled
- * command at runtime.
+ * Instructions are added to envPtr to execute the compiled command at
+ * runtime.
*
*----------------------------------------------------------------------
*/
@@ -4664,9 +4692,10 @@ CompileAssociativeBinaryOpCmd(
}
if (parsePtr->numWords == 2) {
/*
- * TODO: Fixup the single argument case to require
- * numeric argument. Fallback on direct eval until fixed
+ * TODO: Fixup the single argument case to require numeric argument.
+ * Fallback on direct eval until fixed.
*/
+
return TCL_ERROR;
}
for (words=1 ; words<parsePtr->numWords ; words++) {
@@ -4684,16 +4713,16 @@ CompileAssociativeBinaryOpCmd(
*
* CompileStrictlyBinaryOpCmd --
*
- * Utility routine to compile the binary operator commands, that
- * strictly accept exactly two arguments.
+ * Utility routine to compile the binary operator commands, that strictly
+ * accept exactly two arguments.
*
* Results:
* Returns TCL_OK for a successful compile. Returns TCL_ERROR to defer
* evaluation to runtime.
*
* Side effects:
- * Instructions are added to envPtr to execute the compiled
- * command at runtime.
+ * Instructions are added to envPtr to execute the compiled command at
+ * runtime.
*
*----------------------------------------------------------------------
*/
@@ -4811,16 +4840,18 @@ CompileComparisonOpCmd(
*
* TclCompile*OpCmd --
*
- * Procedures called to compile the corresponding
- * "::tcl::mathop::*" commands.
+ * Procedures called to compile the corresponding "::tcl::mathop::*"
+ * commands. These are all wrappers around the utility operator command
+ * compiler functions, except for the compilers for subtraction and
+ * division, which are special.
*
* Results:
* Returns TCL_OK for a successful compile. Returns TCL_ERROR to defer
* evaluation to runtime.
*
* Side effects:
- * Instructions are added to envPtr to execute the compiled
- * command at runtime.
+ * Instructions are added to envPtr to execute the compiled command at
+ * runtime.
*
*----------------------------------------------------------------------
*/
@@ -4842,7 +4873,7 @@ TclCompileNotOpCmd(
{
return CompileUnaryOpCmd(interp, parsePtr, INST_LNOT, envPtr);
}
-
+
int
TclCompileAddOpCmd(
Tcl_Interp *interp,
@@ -4907,96 +4938,8 @@ TclCompilePowOpCmd(
return CompileAssociativeBinaryOpCmd(interp, parsePtr, "1", INST_EXPON,
envPtr);
}
-
-/*
- * This is either clever or stupid.
- *
- * Note the rule: (a-b) = - (b-a)
- * And apply repeatedly to:
- *
- * (((a-b)-c)-d)
- * = - (d - ((a-b)-c))
- * = - (d - - (c - (a-b)))
- * = - (d - - (c - - (b - a)))
- * = - (d + (c + (b - a)))
- * = - ((d + c + b) - a)
- * = (a - (d + c + b))
- *
- * So after word compilation puts the substituted arguments on the stack in
- * reverse order, we don't have to turn them around again and apply repeated
- * INST_SUB instructions. Instead we keep them in reverse order and apply a
- * different sequence of instructions. For N arguments, we apply N-2
- * INST_ADDs, then one INST_SUB. Note that this does the right thing for N=2,
- * a single INST_SUB. When N=1, we can add a phony leading "0" argument and
- * get the right result from the same algorithm as well.
- */
int
-TclCompileMinusOpCmd(
- Tcl_Interp *interp,
- Tcl_Parse *parsePtr,
- CompileEnv *envPtr)
-{
- Tcl_Token *tokenPtr = parsePtr->tokenPtr;
- DefineLineInformation; /* TIP #280 */
- int words;
-
- if (parsePtr->numWords == 1) {
- return TCL_ERROR;
- }
- if (parsePtr->numWords == 2) {
- PushLiteral(envPtr, "0", -1);
- }
- for (words=1 ; words<parsePtr->numWords ; words++) {
- tokenPtr = TokenAfter(tokenPtr);
- CompileWord(envPtr, tokenPtr, interp, words);
- }
- if (parsePtr->numWords == 2) {
- words++;
- }
- while (--words > 2) {
- TclEmitOpcode(INST_ADD, envPtr);
- }
- TclEmitOpcode(INST_SUB, envPtr);
- return TCL_OK;
-}
-
-int
-TclCompileDivOpCmd(
- Tcl_Interp *interp,
- Tcl_Parse *parsePtr,
- CompileEnv *envPtr)
-{
- Tcl_Token *tokenPtr;
- DefineLineInformation; /* TIP #280 */
- int words;
-
- if (parsePtr->numWords == 1) {
- return TCL_ERROR;
- } else if (parsePtr->numWords == 2) {
- PushLiteral(envPtr, "1.0", 3);
- tokenPtr = TokenAfter(parsePtr->tokenPtr);
- CompileWord(envPtr, tokenPtr, interp,1);
- TclEmitOpcode(INST_DIV, envPtr);
- return TCL_OK;
- } else {
- /*
- * TODO: get compiled version that passes mathop-6.18
- * For now, fallback to direct evaluation.
- */
- return TCL_ERROR;
- }
- tokenPtr = TokenAfter(parsePtr->tokenPtr);
- CompileWord(envPtr, tokenPtr, interp,1);
- for (words=2 ; words<parsePtr->numWords ; words++) {
- tokenPtr = TokenAfter(tokenPtr);
- CompileWord(envPtr, tokenPtr, interp,words);
- TclEmitOpcode(INST_DIV, envPtr);
- }
- return TCL_OK;
-}
-
-int
TclCompileLshiftOpCmd(
Tcl_Interp *interp,
Tcl_Parse *parsePtr,
@@ -5059,7 +5002,7 @@ TclCompileNiOpCmd(
return CompileStrictlyBinaryOpCmd(interp, parsePtr, INST_LIST_NOT_IN,
envPtr);
}
-
+
int
TclCompileLessOpCmd(
Tcl_Interp *interp,
@@ -5115,6 +5058,95 @@ TclCompileStreqOpCmd(
}
/*
+ * This is either clever or stupid.
+ *
+ * Note the rule: (a-b) = - (b-a)
+ * And apply repeatedly to:
+ *
+ * (((a-b)-c)-d)
+ * = - (d - ((a-b)-c))
+ * = - (d - - (c - (a-b)))
+ * = - (d - - (c - - (b - a)))
+ * = - (d + (c + (b - a)))
+ * = - ((d + c + b) - a)
+ * = (a - (d + c + b))
+ *
+ * So after word compilation puts the substituted arguments on the stack in
+ * reverse order, we don't have to turn them around again and apply repeated
+ * INST_SUB instructions. Instead we keep them in reverse order and apply a
+ * different sequence of instructions. For N arguments, we apply N-2
+ * INST_ADDs, then one INST_SUB. Note that this does the right thing for N=2,
+ * a single INST_SUB. When N=1, we can add a phony leading "0" argument and
+ * get the right result from the same algorithm as well.
+ */
+
+int
+TclCompileMinusOpCmd(
+ Tcl_Interp *interp,
+ Tcl_Parse *parsePtr,
+ CompileEnv *envPtr)
+{
+ Tcl_Token *tokenPtr = parsePtr->tokenPtr;
+ DefineLineInformation; /* TIP #280 */
+ int words;
+
+ if (parsePtr->numWords == 1) {
+ return TCL_ERROR;
+ }
+ if (parsePtr->numWords == 2) {
+ PushLiteral(envPtr, "0", -1);
+ }
+ for (words=1 ; words<parsePtr->numWords ; words++) {
+ tokenPtr = TokenAfter(tokenPtr);
+ CompileWord(envPtr, tokenPtr, interp, words);
+ }
+ if (parsePtr->numWords == 2) {
+ words++;
+ }
+ while (--words > 2) {
+ TclEmitOpcode(INST_ADD, envPtr);
+ }
+ TclEmitOpcode(INST_SUB, envPtr);
+ return TCL_OK;
+}
+
+int
+TclCompileDivOpCmd(
+ Tcl_Interp *interp,
+ Tcl_Parse *parsePtr,
+ CompileEnv *envPtr)
+{
+ Tcl_Token *tokenPtr;
+ DefineLineInformation; /* TIP #280 */
+ int words;
+
+ if (parsePtr->numWords == 1) {
+ return TCL_ERROR;
+ } else if (parsePtr->numWords == 2) {
+ PushLiteral(envPtr, "1.0", 3);
+ tokenPtr = TokenAfter(parsePtr->tokenPtr);
+ CompileWord(envPtr, tokenPtr, interp,1);
+ TclEmitOpcode(INST_DIV, envPtr);
+ return TCL_OK;
+ } else {
+ /*
+ * TODO: get compiled version that passes mathop-6.18. For now,
+ * fallback to direct evaluation.
+ */
+
+ return TCL_ERROR;
+ }
+ tokenPtr = TokenAfter(parsePtr->tokenPtr);
+ CompileWord(envPtr, tokenPtr, interp,1);
+ for (words=2 ; words<parsePtr->numWords ; words++) {
+ tokenPtr = TokenAfter(tokenPtr);
+ CompileWord(envPtr, tokenPtr, interp,words);
+ TclEmitOpcode(INST_DIV, envPtr);
+ }
+ return TCL_OK;
+}
+
+/*
* Local Variables:
* mode: c
* c-basic-offset: 4