diff options
Diffstat (limited to 'generic/tclCompile.c')
| -rw-r--r-- | generic/tclCompile.c | 1959 | 
1 files changed, 1437 insertions, 522 deletions
| diff --git a/generic/tclCompile.c b/generic/tclCompile.c index 5565342..347e3f0 100644 --- a/generic/tclCompile.c +++ b/generic/tclCompile.c @@ -14,6 +14,7 @@  #include "tclInt.h"  #include "tclCompile.h" +#include <assert.h>  /*   * Table of all AuxData types. @@ -37,7 +38,7 @@ TCL_DECLARE_MUTEX(tableMutex)  int tclTraceCompile = 0;  static int traceInitialized = 0;  #endif - +  /*   * A table describing the Tcl bytecode instructions. Entries in this table   * must correspond to the instruction opcode definitions in tclCompile.h. The @@ -62,7 +63,7 @@ InstructionDesc const tclInstructionTable[] = {  	/* Pop the topmost stack object */      {"dup",		  1,   +1,         0,	{OPERAND_NONE}},  	/* Duplicate the topmost stack object and push the result */ -    {"concat1",		  2,   INT_MIN,    1,	{OPERAND_UINT1}}, +    {"strcat",		  2,   INT_MIN,    1,	{OPERAND_UINT1}},  	/* Concatenate the top op1 items and push result */      {"invokeStk1",	  2,   INT_MIN,    1,	{OPERAND_UINT1}},  	/* Invoke command named objv[0]; <objc,objv> = <op1,top op1> */ @@ -309,7 +310,7 @@ InstructionDesc const tclInstructionTable[] = {      {"pushReturnOpts",	  1,	+1,	   0,	{OPERAND_NONE}},  	/* Push the interpreter's return option dictionary as an object on the  	 * stack. */ -    {"returnStk",	  1,	-2,	   0,	{OPERAND_NONE}}, +    {"returnStk",	  1,	-1,	   0,	{OPERAND_NONE}},  	/* Compiled [return]; options and result are on the stack, code and  	 * level are in the options. */ @@ -372,17 +373,18 @@ InstructionDesc const tclInstructionTable[] = {  	 * Stack:  ... value => ...  	 * Note that the jump table contains offsets relative to the PC when  	 * it points to this instruction; the code is relocatable. */ -    {"upvar",            5,     0,        1,   {OPERAND_LVT4}}, +    {"upvar",            5,    -1,        1,   {OPERAND_LVT4}},  	/* finds level and otherName in stack, links to local variable at  	 * index op1. Leaves the level on stack. */ -    {"nsupvar",          5,     0,        1,   {OPERAND_LVT4}}, +    {"nsupvar",          5,    -1,        1,   {OPERAND_LVT4}},  	/* finds namespace and otherName in stack, links to local variable at  	 * index op1. Leaves the namespace on stack. */ -    {"variable",         5,     0,        1,   {OPERAND_LVT4}}, +    {"variable",         5,    -1,        1,   {OPERAND_LVT4}},  	/* finds namespace and otherName in stack, links to local variable at  	 * index op1. Leaves the namespace on stack. */      {"syntax",		 9,   -1,         2,	{OPERAND_INT4, OPERAND_UINT4}}, -	/* Compiled bytecodes to signal syntax error. */ +	/* Compiled bytecodes to signal syntax error. Equivalent to returnImm +	 * except for the ERR_ALREADY_LOGGED flag in the interpreter. */      {"reverse",		 5,    0,         1,	{OPERAND_UINT4}},  	/* Reverse the order of the arg elements at the top of stack */ @@ -421,6 +423,233 @@ InstructionDesc const tclInstructionTable[] = {  	/* Make general variable cease to exist; unparsed variable name is  	 * stktop; op1 is 1 for errors on problems, 0 otherwise */ +    {"dictExpand",       1,    -1,        0,    {OPERAND_NONE}}, +        /* Probe into a dict and extract it (or a subdict of it) into +         * variables with matched names. Produces list of keys bound as +         * result. Part of [dict with]. +	 * Stack:  ... dict path => ... keyList */ +    {"dictRecombineStk", 1,    -3,        0,    {OPERAND_NONE}}, +        /* Map variable contents back into a dictionary in a variable. Part of +         * [dict with]. +	 * Stack:  ... dictVarName path keyList => ... */ +    {"dictRecombineImm", 5,    -2,        1,    {OPERAND_LVT4}}, +        /* Map variable contents back into a dictionary in the local variable +         * indicated by the LVT index. Part of [dict with]. +	 * Stack:  ... path keyList => ... */ +    {"dictExists",	 5,	INT_MIN,  1,	{OPERAND_UINT4}}, +	/* The top op4 words (min 1) are a key path into the dictionary just +	 * below the keys on the stack, and all those values are replaced by a +	 * boolean indicating whether it is possible to read out a value from +	 * that key-path (like [dict exists]). +	 * Stack:  ... dict key1 ... keyN => ... boolean */ +    {"verifyDict",	 1,    -1,	  0,	{OPERAND_NONE}}, +	/* Verifies that the word on the top of the stack is a dictionary, +	 * popping it if it is and throwing an error if it is not. +	 * Stack:  ... value => ... */ + +    {"strmap",		 1,    -2,	  0,	{OPERAND_NONE}}, +	/* Simplified version of [string map] that only applies one change +	 * string, and only case-sensitively. +	 * Stack:  ... from to string => ... changedString */ +    {"strfind",		 1,    -1,	  0,	{OPERAND_NONE}}, +	/* Find the first index of a needle string in a haystack string, +	 * producing the index (integer) or -1 if nothing found. +	 * Stack:  ... needle haystack => ... index */ +    {"strrfind",	 1,    -1,	  0,	{OPERAND_NONE}}, +	/* Find the last index of a needle string in a haystack string, +	 * producing the index (integer) or -1 if nothing found. +	 * Stack:  ... needle haystack => ... index */ +    {"strrangeImm",	 9,	0,	  2,	{OPERAND_IDX4, OPERAND_IDX4}}, +	/* String Range: push (string range stktop op4 op4) */ +    {"strrange",	 1,    -2,	  0,	{OPERAND_NONE}}, +	/* String Range with non-constant arguments. +	 * Stack:  ... string idxA idxB => ... substring */ + +    {"yield",		 1,	0,	  0,	{OPERAND_NONE}}, +	/* Makes the current coroutine yield the value at the top of the +	 * stack, and places the response back on top of the stack when it +	 * resumes. +	 * Stack:  ... valueToYield => ... resumeValue */ +    {"coroName",         1,    +1,	  0,	{OPERAND_NONE}}, +	/* Push the name of the interpreter's current coroutine as an object +	 * on the stack. */ +    {"tailcall",	 2,    INT_MIN,	  1,	{OPERAND_UINT1}}, +	/* Do a tailcall with the opnd items on the stack as the thing to +	 * tailcall to; opnd must be greater than 0 for the semantics to work +	 * right. */ + +    {"currentNamespace", 1,    +1,	  0,	{OPERAND_NONE}}, +	/* Push the name of the interpreter's current namespace as an object +	 * on the stack. */ +    {"infoLevelNumber",  1,    +1,	  0,	{OPERAND_NONE}}, +	/* Push the stack depth (i.e., [info level]) of the interpreter as an +	 * object on the stack. */ +    {"infoLevelArgs",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Push the argument words to a stack depth (i.e., [info level <n>]) +	 * of the interpreter as an object on the stack. +	 * Stack:  ... depth => ... argList */ +    {"resolveCmd",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Resolves the command named on the top of the stack to its fully +	 * qualified version, or produces the empty string if no such command +	 * exists. Never generates errors. +	 * Stack:  ... cmdName => ... fullCmdName */ + +    {"tclooSelf",	 1,	+1,	  0,	{OPERAND_NONE}}, +	/* Push the identity of the current TclOO object (i.e., the name of +	 * its current public access command) on the stack. */ +    {"tclooClass",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Push the class of the TclOO object named at the top of the stack +	 * onto the stack. +	 * Stack:  ... object => ... class */ +    {"tclooNamespace",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Push the namespace of the TclOO object named at the top of the +	 * stack onto the stack. +	 * Stack:  ... object => ... namespace */ +    {"tclooIsObject",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Push whether the value named at the top of the stack is a TclOO +	 * object (i.e., a boolean). Can corrupt the interpreter result +	 * despite not throwing, so not safe for use in a post-exception +	 * context. +	 * Stack:  ... value => ... boolean */ + +    {"arrayExistsStk",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Looks up the element on the top of the stack and tests whether it +	 * is an array. Pushes a boolean describing whether this is the +	 * case. Also runs the whole-array trace on the named variable, so can +	 * throw anything. +	 * Stack:  ... varName => ... boolean */ +    {"arrayExistsImm",	 5,	+1,	  1,	{OPERAND_UINT4}}, +	/* Looks up the variable indexed by opnd and tests whether it is an +	 * array. Pushes a boolean describing whether this is the case. Also +	 * runs the whole-array trace on the named variable, so can throw +	 * anything. +	 * Stack:  ... => ... boolean */ +    {"arrayMakeStk",	 1,	-1,	  0,	{OPERAND_NONE}}, +	/* Forces the element on the top of the stack to be the name of an +	 * array. +	 * Stack:  ... varName => ... */ +    {"arrayMakeImm",	 5,	0,	  1,	{OPERAND_UINT4}}, +	/* Forces the variable indexed by opnd to be an array. Does not touch +	 * the stack. */ + +    {"invokeReplace",	 6,	INT_MIN,  2,	{OPERAND_UINT4,OPERAND_UINT1}}, +	/* Invoke command named objv[0], replacing the first two words with +	 * the word at the top of the stack; +	 * <objc,objv> = <op4,top op4 after popping 1> */ + +    {"listConcat",	 1,	-1,	  0,	{OPERAND_NONE}}, +	/* Concatenates the two lists at the top of the stack into a single +	 * list and pushes that resulting list onto the stack. +	 * Stack: ... list1 list2 => ... [lconcat list1 list2] */ + +    {"expandDrop",       1,    0,          0,	{OPERAND_NONE}}, +	/* Drops an element from the auxiliary stack, popping stack elements +	 * until the matching stack depth is reached. */ + +    /* New foreach implementation */ +    {"foreach_start",	 5,	+2,	  1,	{OPERAND_AUX4}}, +	/* Initialize execution of a foreach loop. Operand is aux data index +	 * of the ForeachInfo structure for the foreach command. It pushes 2 +	 * elements which hold runtime params for foreach_step, they are later +	 * dropped by foreach_end together with the value lists. NOTE that the +	 * iterator-tracker and info reference must not be passed to bytecodes +	 * that handle normal Tcl values. NOTE that this instruction jumps to +	 * the foreach_step instruction paired with it; the stack info below +	 * is only nominal. +	 * Stack: ... listObjs... => ... listObjs... iterTracker info */ +    {"foreach_step",	 1,	 0,	  0,	{OPERAND_NONE}}, +	/* "Step" or begin next iteration of foreach loop. Assigns to foreach +	 * iteration variables. May jump to straight after the foreach_start +	 * that pushed the iterTracker and info values. MUST be followed +	 * immediately by a foreach_end. +	 * Stack: ... listObjs... iterTracker info => +	 *				... listObjs... iterTracker info */ +    {"foreach_end",	 1,	 0,	  0,	{OPERAND_NONE}}, +	/* Clean up a foreach loop by dropping the info value, the tracker +	 * value and the lists that were being iterated over. +	 * Stack: ... listObjs... iterTracker info => ... */ +    {"lmap_collect",	 1,	-1,	  0,	{OPERAND_NONE}}, +	/* Appends the value at the top of the stack to the list located on +	 * the stack the "other side" of the foreach-related values. +	 * Stack: ... collector listObjs... iterTracker info value => +	 *			... collector listObjs... iterTracker info */ + +    {"strtrim",		 1,	-1,	  0,	{OPERAND_NONE}}, +	/* [string trim] core: removes the characters (designated by the value +	 * at the top of the stack) from both ends of the string and pushes +	 * the resulting string. +	 * Stack: ... string charset => ... trimmedString */ +    {"strtrimLeft",	 1,	-1,	  0,	{OPERAND_NONE}}, +	/* [string trimleft] core: removes the characters (designated by the +	 * value at the top of the stack) from the left of the string and +	 * pushes the resulting string. +	 * Stack: ... string charset => ... trimmedString */ +    {"strtrimRight",	 1,	-1,	  0,	{OPERAND_NONE}}, +	/* [string trimright] core: removes the characters (designated by the +	 * value at the top of the stack) from the right of the string and +	 * pushes the resulting string. +	 * Stack: ... string charset => ... trimmedString */ + +    {"concatStk",	 5,	INT_MIN,  1,	{OPERAND_UINT4}}, +	/* Wrapper round Tcl_ConcatObj(), used for [concat] and [eval]. opnd +	 * is number of values to concatenate. +	 * Operation:	push concat(stk1 stk2 ... stktop) */ + +    {"strcaseUpper",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* [string toupper] core: converts whole string to upper case using +	 * the default (extended "C" locale) rules. +	 * Stack: ... string => ... newString */ +    {"strcaseLower",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* [string tolower] core: converts whole string to upper case using +	 * the default (extended "C" locale) rules. +	 * Stack: ... string => ... newString */ +    {"strcaseTitle",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* [string totitle] core: converts whole string to upper case using +	 * the default (extended "C" locale) rules. +	 * Stack: ... string => ... newString */ +    {"strreplace",	 1,	-3,	  0,	{OPERAND_NONE}}, +	/* [string replace] core: replaces a non-empty range of one string +	 * with the contents of another. +	 * Stack: ... string fromIdx toIdx replacement => ... newString */ + +    {"originCmd",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Reports which command was the origin (via namespace import chain) +	 * of the command named on the top of the stack. +	 * Stack:  ... cmdName => ... fullOriginalCmdName */ + +    {"tclooNext",	 2,	INT_MIN,  1,	{OPERAND_UINT1}}, +	/* Call the next item on the TclOO call chain, passing opnd arguments +	 * (min 1, max 255, *includes* "next").  The result of the invoked +	 * method implementation will be pushed on the stack in place of the +	 * arguments (similar to invokeStk). +	 * Stack:  ... "next" arg2 arg3 -- argN => ... result */ +    {"tclooNextClass",	 2,	INT_MIN,  1,	{OPERAND_UINT1}}, +	/* Call the following item on the TclOO call chain defined by class +	 * className, passing opnd arguments (min 2, max 255, *includes* +	 * "nextto" and the class name). The result of the invoked method +	 * implementation will be pushed on the stack in place of the +	 * arguments (similar to invokeStk). +	 * Stack:  ... "nextto" className arg3 arg4 -- argN => ... result */ + +    {"yieldToInvoke",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Makes the current coroutine yield the value at the top of the +	 * stack, invoking the given command/args with resolution in the given +	 * namespace (all packed into a list), and places the list of values +	 * that are the response back on top of the stack when it resumes. +	 * Stack:  ... [list ns cmd arg1 ... argN] => ... resumeList */ + +    {"numericType",	 1,	0,	  0,	{OPERAND_NONE}}, +	/* Pushes the numeric type code of the word at the top of the stack. +	 * Stack:  ... value => ... typeCode */ +    {"tryCvtToBoolean",	 1,	+1,	  0,	{OPERAND_NONE}}, +	/* Try converting stktop to boolean if possible. No errors. +	 * Stack:  ... value => ... value isStrictBool */ +    {"strclass",	 2,	0,	  1,	{OPERAND_SCLS1}}, +	/* See if all the characters of the given string are a member of the +	 * specified (by opnd) character class. Note that an empty string will +	 * satisfy the class check (standard definition of "all"). +	 * Stack:  ... stringValue => ... boolean */ +      {NULL, 0, 0, 0, {OPERAND_NONE}}  }; @@ -441,11 +670,15 @@ static void		EnterCmdStartData(CompileEnv *envPtr,  static void		FreeByteCodeInternalRep(Tcl_Obj *objPtr);  static void		FreeSubstCodeInternalRep(Tcl_Obj *objPtr);  static int		GetCmdLocEncodingSize(CompileEnv *envPtr); +static int		IsCompactibleCompileEnv(Tcl_Interp *interp, +			    CompileEnv *envPtr);  #ifdef TCL_COMPILE_STATS  static void		RecordByteCodeStats(ByteCode *codePtr);  #endif /* TCL_COMPILE_STATS */ +static void		RegisterAuxDataType(const AuxDataType *typePtr);  static int		SetByteCodeFromAny(Tcl_Interp *interp,  			    Tcl_Obj *objPtr); +static void		StartExpanding(CompileEnv *envPtr);  static int		FormatInstruction(ByteCode *codePtr,  			    const unsigned char *pc, Tcl_Obj *bufferObj);  static void		PrintSourceToObj(Tcl_Obj *appendObj, @@ -460,6 +693,7 @@ static void		EnterCmdWordData(ExtCmdLoc *eclPtr, int srcOffset,  			    Tcl_Token *tokenPtr, const char *cmd, int len,  			    int numWords, int line, int *clNext, int **lines,  			    CompileEnv *envPtr); +static void		ReleaseCmdWordData(ExtCmdLoc *eclPtr);  /*   * The structure below defines the bytecode Tcl object type by means of @@ -499,6 +733,13 @@ static const Tcl_ObjType tclInstNameType = {      UpdateStringOfInstName,	/* updateStringProc */      NULL,			/* setFromAnyProc */  }; + +/* + * Helper macros. + */ + +#define TclIncrUInt4AtPtr(ptr, delta) \ +    TclStoreInt4AtPtr(TclGetUInt4AtPtr(ptr)+(delta), (ptr));  /*   *---------------------------------------------------------------------- @@ -509,12 +750,13 @@ static const Tcl_ObjType tclInstNameType = {   *	generate an byte code internal form for the Tcl object "objPtr" by   *	compiling its string representation. This function also takes a hook   *	procedure that will be invoked to perform any needed post processing - *	on the compilation results before generating byte codes. + *	on the compilation results before generating byte codes. interp is + *	compilation context and may not be NULL.   *   * Results:   *	The return value is a standard Tcl object result. If an error occurs   *	during compilation, an error message is left in the interpreter's - *	result unless "interp" is NULL. + *	result.   *   * Side effects:   *	Frees the old internal representation. If no error occurs, then the @@ -536,11 +778,9 @@ TclSetByteCodeFromAny(      Interp *iPtr = (Interp *) interp;      CompileEnv compEnv;		/* Compilation environment structure allocated  				 * in frame. */ -    register const AuxData *auxDataPtr; -    LiteralEntry *entryPtr; -    register int i;      int length, result = TCL_OK;      const char *stringPtr; +    Proc *procPtr = iPtr->compiledProcPtr;      ContLineLoc *clLocPtr;  #ifdef TCL_COMPILE_DEBUG @@ -578,9 +818,7 @@ TclSetByteCodeFromAny(      clLocPtr = TclContinuationsGet(objPtr);      if (clLocPtr) { -	compEnv.clLoc = clLocPtr; -	compEnv.clNext = &compEnv.clLoc->loc[0]; -	Tcl_Preserve(compEnv.clLoc); +	compEnv.clNext = &clLocPtr->loc[0];      }      TclCompileScript(interp, stringPtr, length, &compEnv); @@ -592,6 +830,40 @@ TclSetByteCodeFromAny(      TclEmitOpcode(INST_DONE, &compEnv);      /* +     * Check for optimizations! +     * +     * Test if the generated code is free of most hazards; if so, recompile +     * but with generation of INST_START_CMD disabled. This produces somewhat +     * faster code in some cases, and more compact code in more. +     */ + +    if (Tcl_GetMaster(interp) == NULL && +	    !Tcl_LimitTypeEnabled(interp, TCL_LIMIT_COMMANDS|TCL_LIMIT_TIME) +	    && IsCompactibleCompileEnv(interp, &compEnv)) { +	TclFreeCompileEnv(&compEnv); +	iPtr->compiledProcPtr = procPtr; +	TclInitCompileEnv(interp, &compEnv, stringPtr, length, +		iPtr->invokeCmdFramePtr, iPtr->invokeWord); +	if (clLocPtr) { +	    compEnv.clNext = &clLocPtr->loc[0]; +	} +	compEnv.atCmdStart = 2;		/* The disabling magic. */ +	TclCompileScript(interp, stringPtr, length, &compEnv); +	assert (compEnv.atCmdStart > 1); +	TclEmitOpcode(INST_DONE, &compEnv); +	assert (compEnv.atCmdStart > 1); +    } + +    /* +     * Apply some peephole optimizations that can cross specific/generic +     * instruction generator boundaries. +     */ + +    if (iPtr->extra.optimizer) { +	(iPtr->extra.optimizer)(&compEnv); +    } + +    /*       * Invoke the compilation hook procedure if one exists.       */ @@ -608,35 +880,14 @@ TclSetByteCodeFromAny(      TclVerifyLocalLiteralTable(&compEnv);  #endif /*TCL_COMPILE_DEBUG*/ -    TclInitByteCodeObj(objPtr, &compEnv); -#ifdef TCL_COMPILE_DEBUG -    if (tclTraceCompile >= 2) { -	TclPrintByteCodeObj(interp, objPtr); -	fflush(stdout); -    } -#endif /* TCL_COMPILE_DEBUG */ - -    if (result != TCL_OK) { -	/* -	 * Handle any error from the hookProc -	 */ - -	entryPtr = compEnv.literalArrayPtr; -	for (i = 0;  i < compEnv.literalArrayNext;  i++) { -	    TclReleaseLiteral(interp, entryPtr->objPtr); -	    entryPtr++; -	} +    if (result == TCL_OK) { +	TclInitByteCodeObj(objPtr, &compEnv);  #ifdef TCL_COMPILE_DEBUG -	TclVerifyGlobalLiteralTable(iPtr); -#endif /*TCL_COMPILE_DEBUG*/ - -	auxDataPtr = compEnv.auxDataArrayPtr; -	for (i = 0;  i < compEnv.auxDataArrayNext;  i++) { -	    if (auxDataPtr->type->freeProc != NULL) { -		auxDataPtr->type->freeProc(auxDataPtr->clientData); -	    } -	    auxDataPtr++; +	if (tclTraceCompile >= 2) { +	    TclPrintByteCodeObj(interp, objPtr); +	    fflush(stdout);  	} +#endif /* TCL_COMPILE_DEBUG */      }      TclFreeCompileEnv(&compEnv); @@ -672,8 +923,10 @@ SetByteCodeFromAny(  				 * compiled. Must not be NULL. */      Tcl_Obj *objPtr)		/* The object to make a ByteCode object. */  { -    TclSetByteCodeFromAny(interp, objPtr, NULL, NULL); -    return TCL_OK; +    if (interp == NULL) { +	return TCL_ERROR; +    } +    return TclSetByteCodeFromAny(interp, objPtr, NULL, NULL);  }  /* @@ -727,10 +980,9 @@ static void  FreeByteCodeInternalRep(      register Tcl_Obj *objPtr)	/* Object whose internal rep to free. */  { -    register ByteCode *codePtr = objPtr->internalRep.otherValuePtr; +    register ByteCode *codePtr = objPtr->internalRep.twoPtrValue.ptr1;      objPtr->typePtr = NULL; -    objPtr->internalRep.otherValuePtr = NULL;      codePtr->refCount--;      if (codePtr->refCount <= 0) {  	TclCleanupByteCode(codePtr); @@ -750,9 +1002,8 @@ FreeByteCodeInternalRep(   *	None.   *   * Side effects: - *	Frees objPtr's bytecode internal representation and sets its type and - *	objPtr->internalRep.otherValuePtr NULL. Also releases its literals and - *	frees its auxiliary data items. + *	Frees objPtr's bytecode internal representation and sets its type NULL + *	Also releases its literals and frees its auxiliary data items.   *   *----------------------------------------------------------------------   */ @@ -827,7 +1078,7 @@ TclCleanupByteCode(       * released.       */ -    if ((codePtr->flags & TCL_BYTECODE_PRECOMPILED) || (interp == NULL)) { +    if (codePtr->flags & TCL_BYTECODE_PRECOMPILED) {  	objArrayPtr = codePtr->objArrayPtr;  	for (i = 0;  i < numLitObjects;  i++) { @@ -840,17 +1091,9 @@ TclCleanupByteCode(  	codePtr->numLitObjects = 0;      } else {  	objArrayPtr = codePtr->objArrayPtr; -	for (i = 0;  i < numLitObjects;  i++) { -	    /* -	     * TclReleaseLiteral sets a ByteCode's object array entry NULL to -	     * indicate that it has already freed the literal. -	     */ - -	    objPtr = *objArrayPtr; -	    if (objPtr != NULL) { -		TclReleaseLiteral(interp, objPtr); -	    } -	    objArrayPtr++; +	while (numLitObjects--) { +	    /* TclReleaseLiteral calls Tcl_DecrRefCount() for us */ +	    TclReleaseLiteral(interp, *objArrayPtr++);  	}      } @@ -875,22 +1118,7 @@ TclCleanupByteCode(  		(char *) codePtr);  	if (hePtr) { -	    ExtCmdLoc *eclPtr = Tcl_GetHashValue(hePtr); - -	    if (eclPtr->type == TCL_LOCATION_SOURCE) { -		Tcl_DecrRefCount(eclPtr->path); -	    } -	    for (i=0 ; i<eclPtr->nuloc ; i++) { -		ckfree(eclPtr->loc[i].line); -	    } - -	    if (eclPtr->loc != NULL) { -		ckfree(eclPtr->loc); -	    } - -	    Tcl_DeleteHashTable(&eclPtr->litInfo); - -	    ckfree(eclPtr); +	    ReleaseCmdWordData(Tcl_GetHashValue(hePtr));  	    Tcl_DeleteHashEntry(hePtr);  	}      } @@ -904,6 +1132,77 @@ TclCleanupByteCode(  }  /* + * --------------------------------------------------------------------- + * + * IsCompactibleCompileEnv -- + * + *	Checks to see if we may apply some basic compaction optimizations to a + *	piece of bytecode. Idempotent. + * + * --------------------------------------------------------------------- + */ + +static int +IsCompactibleCompileEnv( +    Tcl_Interp *interp, +    CompileEnv *envPtr) +{ +    unsigned char *pc; +    int size; + +    /* +     * Special: procedures in the '::tcl' namespace (or its children) are +     * considered to be well-behaved and so can have compaction applied even +     * if it would otherwise be invalid. +     */ + +    if (envPtr->procPtr != NULL && envPtr->procPtr->cmdPtr != NULL +	    && envPtr->procPtr->cmdPtr->nsPtr != NULL) { +	Namespace *nsPtr = envPtr->procPtr->cmdPtr->nsPtr; + +	if (strcmp(nsPtr->fullName, "::tcl") == 0 +		|| strncmp(nsPtr->fullName, "::tcl::", 7) == 0) { +	    return 1; +	} +    } + +    /* +     * Go through and ensure that no operation involved can cause a desired +     * change of bytecode sequence during running. This comes down to ensuring +     * that there are no mapped variables (due to traces) or calls to external +     * commands (traces, [uplevel] trickery). This is actually a very +     * conservative check; it turns down a lot of code that is OK in practice. +     */ + +    for (pc = envPtr->codeStart ; pc < envPtr->codeNext ; pc += size) { +	switch (*pc) { +	    /* Invokes */ +	case INST_INVOKE_STK1: +	case INST_INVOKE_STK4: +	case INST_INVOKE_EXPANDED: +	case INST_INVOKE_REPLACE: +	    return 0; +	    /* Runtime evals */ +	case INST_EVAL_STK: +	case INST_EXPR_STK: +	case INST_YIELD: +	    return 0; +	    /* Upvars */ +	case INST_UPVAR: +	case INST_NSUPVAR: +	case INST_VARIABLE: +	    return 0; +	default: +	    size = tclInstructionTable[*pc].numBytes; +	    assert (size > 0); +	    break; +	} +    } + +    return 1; +} + +/*   *----------------------------------------------------------------------   *   * Tcl_SubstObj -- @@ -1027,14 +1326,19 @@ CompileSubstObj(  	objPtr->typePtr = &substCodeType;  	TclFreeCompileEnv(&compEnv); -	codePtr = objPtr->internalRep.otherValuePtr; +	codePtr = objPtr->internalRep.twoPtrValue.ptr1;  	objPtr->internalRep.ptrAndLongRep.ptr = codePtr;  	objPtr->internalRep.ptrAndLongRep.value = flags;  	if (iPtr->varFramePtr->localCachePtr) {  	    codePtr->localCachePtr = iPtr->varFramePtr->localCachePtr;  	    codePtr->localCachePtr->refCount++;  	} -	/* TODO: Debug printing? */ +#ifdef TCL_COMPILE_DEBUG +	if (tclTraceCompile >= 2) { +	    TclPrintByteCodeObj(interp, objPtr); +	    fflush(stdout); +	} +#endif /* TCL_COMPILE_DEBUG */      }      return codePtr;  } @@ -1066,12 +1370,31 @@ FreeSubstCodeInternalRep(      register ByteCode *codePtr = objPtr->internalRep.ptrAndLongRep.ptr;      objPtr->typePtr = NULL; -    objPtr->internalRep.otherValuePtr = NULL;      codePtr->refCount--;      if (codePtr->refCount <= 0) {  	TclCleanupByteCode(codePtr);      }  } + +static void +ReleaseCmdWordData( +    ExtCmdLoc *eclPtr) +{ +    int i; + +    if (eclPtr->type == TCL_LOCATION_SOURCE) { +	Tcl_DecrRefCount(eclPtr->path); +    } +    for (i=0 ; i<eclPtr->nuloc ; i++) { +	ckfree((char *) eclPtr->loc[i].line); +    } + +    if (eclPtr->loc != NULL) { +	ckfree((char *) eclPtr->loc); +    } + +    ckfree((char *) eclPtr); +}  /*   *---------------------------------------------------------------------- @@ -1104,6 +1427,8 @@ TclInitCompileEnv(  {      Interp *iPtr = (Interp *) interp; +    assert(tclInstructionTable[LAST_INST_OPCODE+1].name == NULL); +      envPtr->iPtr = iPtr;      envPtr->source = stringPtr;      envPtr->numSrcBytes = numBytes; @@ -1127,6 +1452,7 @@ TclInitCompileEnv(      envPtr->mallocedLiteralArray = 0;      envPtr->exceptArrayPtr = envPtr->staticExceptArraySpace; +    envPtr->exceptAuxArrayPtr = envPtr->staticExAuxArraySpace;      envPtr->exceptArrayNext = 0;      envPtr->exceptArrayEnd = COMPILEENV_INIT_EXCEPT_RANGES;      envPtr->mallocedExceptArray = 0; @@ -1135,6 +1461,7 @@ TclInitCompileEnv(      envPtr->cmdMapEnd = COMPILEENV_INIT_CMD_MAP_SIZE;      envPtr->mallocedCmdMap = 0;      envPtr->atCmdStart = 1; +    envPtr->expandCount = 0;      /*       * TIP #280: Set up the extended command location information, based on @@ -1150,9 +1477,8 @@ TclInitCompileEnv(      envPtr->extCmdMapPtr->nloc = 0;      envPtr->extCmdMapPtr->nuloc = 0;      envPtr->extCmdMapPtr->path = NULL; -    Tcl_InitHashTable(&envPtr->extCmdMapPtr->litInfo, TCL_ONE_WORD_KEYS); -    if ((invoker == NULL) || (invoker->type == TCL_LOCATION_EVAL_LIST)) { +    if (invoker == NULL) {  	/*  	 * Initialize the compiler for relative counting in case of a  	 * dynamic context. @@ -1266,7 +1592,6 @@ TclInitCompileEnv(       * data is available.       */ -    envPtr->clLoc = NULL;      envPtr->clNext = NULL;      envPtr->auxDataArrayPtr = envPtr->staticAuxDataArraySpace; @@ -1305,6 +1630,32 @@ TclFreeCompileEnv(  	ckfree(envPtr->localLitTable.buckets);  	envPtr->localLitTable.buckets = envPtr->localLitTable.staticBuckets;      } +    if (envPtr->iPtr) { +	/*  +	 * We never converted to Bytecode, so free the things we would +	 * have transferred to it. +	 */ + +	int i; +	LiteralEntry *entryPtr = envPtr->literalArrayPtr; +	AuxData *auxDataPtr = envPtr->auxDataArrayPtr; + +	for (i = 0;  i < envPtr->literalArrayNext;  i++) { +	    TclReleaseLiteral((Tcl_Interp *)envPtr->iPtr, entryPtr->objPtr); +	    entryPtr++; +	} + +#ifdef TCL_COMPILE_DEBUG +	TclVerifyGlobalLiteralTable(envPtr->iPtr); +#endif /*TCL_COMPILE_DEBUG*/ + +	for (i = 0;  i < envPtr->auxDataArrayNext;  i++) { +	    if (auxDataPtr->type->freeProc != NULL) { +		auxDataPtr->type->freeProc(auxDataPtr->clientData); +	    } +	    auxDataPtr++; +	} +    }      if (envPtr->mallocedCodeArray) {  	ckfree(envPtr->codeStart);      } @@ -1313,6 +1664,7 @@ TclFreeCompileEnv(      }      if (envPtr->mallocedExceptArray) {  	ckfree(envPtr->exceptArrayPtr); +	ckfree(envPtr->exceptAuxArrayPtr);      }      if (envPtr->mallocedCmdMap) {  	ckfree(envPtr->cmdMapPtr); @@ -1321,17 +1673,8 @@ TclFreeCompileEnv(  	ckfree(envPtr->auxDataArrayPtr);      }      if (envPtr->extCmdMapPtr) { -	ckfree(envPtr->extCmdMapPtr); -    } - -    /* -     * If we used data about invisible continuation lines, then now is the -     * time to release on our hold on it. The lock was set in function -     * TclSetByteCodeFromAny(), found in this file. -     */ - -    if (envPtr->clLoc) { -	Tcl_Release(envPtr->clLoc); +	ReleaseCmdWordData(envPtr->extCmdMapPtr); +	envPtr->extCmdMapPtr = NULL;      }  } @@ -1433,452 +1776,467 @@ TclWordKnownAtCompileTime(   *----------------------------------------------------------------------   */ +static int +ExpandRequested( +    Tcl_Token *tokenPtr, +    int numWords) +{ +    /* Determine whether any words of the command require expansion */ +    while (numWords--) { +	if (tokenPtr->type == TCL_TOKEN_EXPAND_WORD) { +	    return 1; +	} +	tokenPtr = TokenAfter(tokenPtr); +    } +    return 0; +} + +static void +CompileCmdLiteral( +    Tcl_Interp *interp, +    Tcl_Obj *cmdObj, +    CompileEnv *envPtr) +{ +    int numBytes; +    const char *bytes = Tcl_GetStringFromObj(cmdObj, &numBytes); +    int cmdLitIdx = TclRegisterNewCmdLiteral(envPtr, bytes, numBytes); +    Command *cmdPtr = (Command *) Tcl_GetCommandFromObj(interp, cmdObj); + +    if (cmdPtr) { +	TclSetCmdNameObj(interp, TclFetchLiteral(envPtr, cmdLitIdx), cmdPtr); +    } +    TclEmitPush(cmdLitIdx, envPtr); +} +  void -TclCompileScript( -    Tcl_Interp *interp,		/* Used for error and status reporting. Also -				 * serves as context for finding and compiling -				 * commands. May not be NULL. */ -    const char *script,		/* The source script to compile. */ -    int numBytes,		/* Number of bytes in script. If < 0, the -				 * script consists of all bytes up to the -				 * first null character. */ -    CompileEnv *envPtr)		/* Holds resulting instructions. */ +TclCompileInvocation( +    Tcl_Interp *interp, +    Tcl_Token *tokenPtr, +    Tcl_Obj *cmdObj, +    int numWords, +    CompileEnv *envPtr)  { -    Interp *iPtr = (Interp *) interp; -    int lastTopLevelCmdIndex = -1; -				/* Index of most recent toplevel command in -				 * the command location table. Initialized to -				 * avoid compiler warning. */ -    int startCodeOffset = -1;	/* Offset of first byte of current command's -				 * code. Init. to avoid compiler warning. */ -    unsigned char *entryCodeNext = envPtr->codeNext; -    const char *p, *next; -    Namespace *cmdNsPtr; -    Command *cmdPtr; -    Tcl_Token *tokenPtr; -    int bytesLeft, isFirstCmd, wordIdx, currCmdIndex, commandLength, objIndex; -    Tcl_DString ds; -    /* TIP #280 */ -    ExtCmdLoc *eclPtr = envPtr->extCmdMapPtr; -    int *wlines, wlineat, cmdLine, *clNext; -    Tcl_Parse *parsePtr = TclStackAlloc(interp, sizeof(Tcl_Parse)); +    int wordIdx = 0, depth = TclGetStackDepth(envPtr); +    DefineLineInformation; + +    if (cmdObj) { +	CompileCmdLiteral(interp, cmdObj, envPtr); +	wordIdx = 1; +	tokenPtr = TokenAfter(tokenPtr); +    } -    Tcl_DStringInit(&ds); +    for (; wordIdx < numWords; wordIdx++, tokenPtr = TokenAfter(tokenPtr)) { +	int objIdx; -    if (numBytes < 0) { -	numBytes = strlen(script); +	SetLineInformation(wordIdx); + +	if (tokenPtr->type != TCL_TOKEN_SIMPLE_WORD) { +	    CompileTokens(envPtr, tokenPtr, interp); +	    continue; +	} + +	objIdx = TclRegisterNewLiteral(envPtr, +		tokenPtr[1].start, tokenPtr[1].size); +	if (envPtr->clNext) { +	    TclContinuationsEnterDerived(TclFetchLiteral(envPtr, objIdx), +		    tokenPtr[1].start - envPtr->source, envPtr->clNext); +	} +	TclEmitPush(objIdx, envPtr);      } -    Tcl_ResetResult(interp); -    isFirstCmd = 1; -    if (envPtr->procPtr != NULL) { -	cmdNsPtr = envPtr->procPtr->cmdPtr->nsPtr; +    if (wordIdx <= 255) { +	TclEmitInvoke(envPtr, INST_INVOKE_STK1, wordIdx);      } else { -	cmdNsPtr = NULL;	/* use current NS */ +	TclEmitInvoke(envPtr, INST_INVOKE_STK4, wordIdx);      } +    TclCheckStackDepth(depth+1, envPtr); +} -    /* -     * Each iteration through the following loop compiles the next command -     * from the script. -     */ +static void +CompileExpanded( +    Tcl_Interp *interp, +    Tcl_Token *tokenPtr, +    Tcl_Obj *cmdObj, +    int numWords, +    CompileEnv *envPtr) +{ +    int wordIdx = 0; +    DefineLineInformation; +    int depth = TclGetStackDepth(envPtr); +     +    StartExpanding(envPtr); +    if (cmdObj) { +	CompileCmdLiteral(interp, cmdObj, envPtr); +	wordIdx = 1; +	tokenPtr = TokenAfter(tokenPtr); +    } -    p = script; -    bytesLeft = numBytes; -    cmdLine = envPtr->line; -    clNext = envPtr->clNext; -    do { -	if (Tcl_ParseCommand(interp, p, bytesLeft, 0, parsePtr) != TCL_OK) { -	    /* -	     * Compile bytecodes to report the parse error at runtime. -	     */ +    for (; wordIdx < numWords; wordIdx++, tokenPtr = TokenAfter(tokenPtr)) { +	int objIdx; -	    Tcl_LogCommandInfo(interp, script, parsePtr->commandStart, -		    /* Drop the command terminator (";","]") if appropriate */ -		    (parsePtr->term == -		    parsePtr->commandStart + parsePtr->commandSize - 1)? -		    parsePtr->commandSize - 1 : parsePtr->commandSize); -	    TclCompileSyntaxError(interp, envPtr); -	    break; +	SetLineInformation(wordIdx); + +	if (tokenPtr->type != TCL_TOKEN_SIMPLE_WORD) { +	    CompileTokens(envPtr, tokenPtr, interp); +	    if (tokenPtr->type == TCL_TOKEN_EXPAND_WORD) { +		TclEmitInstInt4(INST_EXPAND_STKTOP, +			envPtr->currStackDepth, envPtr); +	    } +	    continue;  	} -	/* -	 * TIP #280: We have to count newlines before the command even in the -	 * degenerate case when the command has no words. (See test -	 * info-30.33). -	 * So make that counting here, and not in the (numWords > 0) branch -	 * below. -	 */ +	objIdx = TclRegisterNewLiteral(envPtr, +		tokenPtr[1].start, tokenPtr[1].size); +	if (envPtr->clNext) { +	    TclContinuationsEnterDerived(TclFetchLiteral(envPtr, objIdx), +		    tokenPtr[1].start - envPtr->source, envPtr->clNext); +	} +	TclEmitPush(objIdx, envPtr); +    } -	TclAdvanceLines(&cmdLine, p, parsePtr->commandStart); -	TclAdvanceContinuations(&cmdLine, &clNext, -		parsePtr->commandStart - envPtr->source); +    /* +     * The stack depth during argument expansion can only be managed at +     * runtime, as the number of elements in the expanded lists is not known +     * at compile time. We adjust here the stack depth estimate so that it is +     * correct after the command with expanded arguments returns. +     * +     * The end effect of this command's invocation is that all the words of +     * the command are popped from the stack, and the result is pushed: the +     * stack top changes by (1-wordIdx). +     * +     * Note that the estimates are not correct while the command is being +     * prepared and run, INST_EXPAND_STKTOP is not stack-neutral in general. +     */ -	if (parsePtr->numWords > 0) { -	    int expand = 0;	/* Set if there are dynamic expansions to -				 * handle */ +    TclEmitInvoke(envPtr, INST_INVOKE_EXPANDED, wordIdx); +    TclCheckStackDepth(depth+1, envPtr); +} -	    /* -	     * If not the first command, pop the previous command's result -	     * and, if we're compiling a top level command, update the last -	     * command's code size to account for the pop instruction. -	     */ +static int  +CompileCmdCompileProc( +    Tcl_Interp *interp, +    Tcl_Parse *parsePtr, +    Command *cmdPtr, +    CompileEnv *envPtr) +{ +    int unwind = 0, incrOffset = -1; +    DefineLineInformation; +    int depth = TclGetStackDepth(envPtr); -	    if (!isFirstCmd) { -		TclEmitOpcode(INST_POP, envPtr); -		envPtr->cmdMapPtr[lastTopLevelCmdIndex].numCodeBytes = -			(envPtr->codeNext - envPtr->codeStart) -			- startCodeOffset; -	    } +    /* +     * Emit of the INST_START_CMD instruction is controlled by the value of +     * envPtr->atCmdStart: +     * +     * atCmdStart == 2	: We are not using the INST_START_CMD instruction. +     * atCmdStart == 1	: INST_START_CMD was the last instruction emitted. +     *			: We do not need to emit another.  Instead we +     *			: increment the number of cmds started at it (except +     *			: for the special case at the start of a script.) +     * atCmdStart == 0	: The last instruction was something else.  We need +     *			: to emit INST_START_CMD here. +     */ + +    switch (envPtr->atCmdStart) { +    case 0: +	unwind = tclInstructionTable[INST_START_CMD].numBytes; +	TclEmitInstInt4(INST_START_CMD, 0, envPtr); +	incrOffset = envPtr->codeNext - envPtr->codeStart; +	TclEmitInt4(0, envPtr); +	break; +    case 1: +	if (envPtr->codeNext > envPtr->codeStart) { +	    incrOffset = envPtr->codeNext - 4 - envPtr->codeStart; +	} +	break; +    case 2: +	/* Nothing to do */ +	; +    } +    if (TCL_OK == TclAttemptCompileProc(interp, parsePtr, 1, cmdPtr, envPtr)) { +	if (incrOffset >= 0) {  	    /* -	     * Determine the actual length of the command. +	     * We successfully compiled a command.  Increment the number of +	     * commands that start at the currently active INST_START_CMD.  	     */ -	    commandLength = parsePtr->commandSize; -	    if (parsePtr->term == parsePtr->commandStart + commandLength-1) { -		/* -		 * The command terminator character (such as ; or ]) is the -		 * last character in the parsed command. Reduce the length by -		 * one so that the trace message doesn't include the -		 * terminator character. -		 */ +	    unsigned char *incrPtr = envPtr->codeStart + incrOffset; +	    unsigned char *startPtr = incrPtr - 5; -		commandLength -= 1; +	    TclIncrUInt4AtPtr(incrPtr, 1); +	    if (unwind) { +		/* We started the INST_START_CMD.  Record the code length. */ +		TclStoreInt4AtPtr(envPtr->codeNext - startPtr, startPtr + 1);  	    } +	} +	TclCheckStackDepth(depth+1, envPtr); +	return TCL_OK; +    } -#ifdef TCL_COMPILE_DEBUG -	    /* -	     * If tracing, print a line for each top level command compiled. -	     */ +    envPtr->codeNext -= unwind; /* Unwind INST_START_CMD */ -	    if ((tclTraceCompile >= 1) && (envPtr->procPtr == NULL)) { -		fprintf(stdout, "  Compiling: "); -		TclPrintSource(stdout, parsePtr->commandStart, -			TclMin(commandLength, 55)); -		fprintf(stdout, "\n"); -	    } -#endif +    /* +     * Throw out any line information generated by the failed compile attempt. +     */ -	    /* -	     * Check whether expansion has been requested for any of the -	     * words. -	     */ +    while (mapPtr->nuloc - 1 > eclIndex) { +	mapPtr->nuloc--; +	ckfree(mapPtr->loc[mapPtr->nuloc].line); +	mapPtr->loc[mapPtr->nuloc].line = NULL; +    } -	    for (wordIdx = 0, tokenPtr = parsePtr->tokenPtr; -		    wordIdx < parsePtr->numWords; -		    wordIdx++, tokenPtr += tokenPtr->numComponents + 1) { -		if (tokenPtr->type == TCL_TOKEN_EXPAND_WORD) { -		    expand = 1; -		    break; -		} -	    } +    /* +     * Reset the index of next command.  Toss out any from failed nested +     * partial compiles. +     */ -	    envPtr->numCommands++; -	    currCmdIndex = envPtr->numCommands - 1; -	    lastTopLevelCmdIndex = currCmdIndex; -	    startCodeOffset = envPtr->codeNext - envPtr->codeStart; -	    EnterCmdStartData(envPtr, currCmdIndex, -		    parsePtr->commandStart - envPtr->source, startCodeOffset); +    envPtr->numCommands = mapPtr->nuloc; +    return TCL_ERROR; +} -	    /* -	     * Should only start issuing instructions after the "command has -	     * started" so that the command range is correct in the bytecode. -	     */ +static int +CompileCommandTokens( +    Tcl_Interp *interp, +    Tcl_Parse *parsePtr, +    CompileEnv *envPtr) +{ +    Interp *iPtr = (Interp *) interp; +    Tcl_Token *tokenPtr = parsePtr->tokenPtr; +    ExtCmdLoc *eclPtr = envPtr->extCmdMapPtr; +    Tcl_Obj *cmdObj = Tcl_NewObj(); +    Command *cmdPtr = NULL; +    int code = TCL_ERROR; +    int cmdKnown, expand = -1; +    int *wlines, wlineat; +    int cmdLine = envPtr->line; +    int *clNext = envPtr->clNext; +    int cmdIdx = envPtr->numCommands; +    int startCodeOffset = envPtr->codeNext - envPtr->codeStart; +    int depth = TclGetStackDepth(envPtr); +     +    assert (parsePtr->numWords > 0); + +    /* Pre-Compile */ + +    envPtr->numCommands++; +    EnterCmdStartData(envPtr, cmdIdx, +	    parsePtr->commandStart - envPtr->source, startCodeOffset); -	    if (expand) { -		TclEmitOpcode(INST_EXPAND_START, envPtr); -	    } +    /* +     * TIP #280. Scan the words and compute the extended location information. +     * The map first contain full per-word line information for use by the +     * compiler. This is later replaced by a reduced form which signals +     * non-literal words, stored in 'wlines'. +     */ -	    /* -	     * TIP #280. Scan the words and compute the extended location -	     * information. The map first contain full per-word line -	     * information for use by the compiler. This is later replaced by -	     * a reduced form which signals non-literal words, stored in -	     * 'wlines'. -	     */ +    EnterCmdWordData(eclPtr, parsePtr->commandStart - envPtr->source, +	    parsePtr->tokenPtr, parsePtr->commandStart, +	    parsePtr->commandSize, parsePtr->numWords, cmdLine, +	    clNext, &wlines, envPtr); +    wlineat = eclPtr->nuloc - 1; -	    EnterCmdWordData(eclPtr, parsePtr->commandStart - envPtr->source, -		    parsePtr->tokenPtr, parsePtr->commandStart, -		    parsePtr->commandSize, parsePtr->numWords, cmdLine, -		    clNext, &wlines, envPtr); -	    wlineat = eclPtr->nuloc - 1; +    envPtr->line = eclPtr->loc[wlineat].line[0]; +    envPtr->clNext = eclPtr->loc[wlineat].next[0]; +    /* Do we know the command word? */ +    Tcl_IncrRefCount(cmdObj); +    tokenPtr = parsePtr->tokenPtr; +    cmdKnown = TclWordKnownAtCompileTime(tokenPtr, cmdObj); + +    /* Is this a command we should (try to) compile with a compileProc ? */ +    if (cmdKnown && !(iPtr->flags & DONT_COMPILE_CMDS_INLINE)) { +	cmdPtr = (Command *) Tcl_GetCommandFromObj(interp, cmdObj); +	if (cmdPtr) {  	    /* -	     * Each iteration of the following loop compiles one word from the -	     * command. +	     * Found a command.  Test the ways we can be told not to attempt +	     * to compile it.  	     */ +	    if ((cmdPtr->compileProc == NULL) +		    || (cmdPtr->nsPtr->flags & NS_SUPPRESS_COMPILATION) +		    || (cmdPtr->flags & CMD_HAS_EXEC_TRACES)) { +		cmdPtr = NULL; +	    } +	} +	if (cmdPtr && !(cmdPtr->flags & CMD_COMPILES_EXPANDED)) { +	    expand = ExpandRequested(parsePtr->tokenPtr, parsePtr->numWords); +	    if (expand) { +		/* We need to expand, but compileProc cannot. */ +		cmdPtr = NULL; +	    } +	} +    } -	    for (wordIdx = 0, tokenPtr = parsePtr->tokenPtr; -		    wordIdx < parsePtr->numWords; wordIdx++, -		    tokenPtr += tokenPtr->numComponents + 1) { +    /* If cmdPtr != NULL, we will try to call cmdPtr->compileProc */ +    if (cmdPtr) { +	code = CompileCmdCompileProc(interp, parsePtr, cmdPtr, envPtr); +    } -		envPtr->line = eclPtr->loc[wlineat].line[wordIdx]; -		envPtr->clNext = eclPtr->loc[wlineat].next[wordIdx]; -		if (tokenPtr->type != TCL_TOKEN_SIMPLE_WORD) { -		    /* -		     * The word is not a simple string of characters. -		     */ +    if (code == TCL_ERROR) { +	if (expand < 0) { +	    expand = ExpandRequested(parsePtr->tokenPtr, parsePtr->numWords); +	} -		    TclCompileTokens(interp, tokenPtr+1, -			    tokenPtr->numComponents, envPtr); -		    if (tokenPtr->type == TCL_TOKEN_EXPAND_WORD) { -			TclEmitInstInt4(INST_EXPAND_STKTOP, -				envPtr->currStackDepth, envPtr); -		    } -		    continue; -		} +	if (expand) { +	    CompileExpanded(interp, parsePtr->tokenPtr, +		    cmdKnown ? cmdObj : NULL, parsePtr->numWords, envPtr); +	} else { +	    TclCompileInvocation(interp, parsePtr->tokenPtr, +		    cmdKnown ? cmdObj : NULL, parsePtr->numWords, envPtr); +	} +    } -		/* -		 * This is a simple string of literal characters (i.e. we know -		 * it absolutely and can use it directly). If this is the -		 * first word and the command has a compile procedure, let it -		 * compile the command. -		 */ +    Tcl_DecrRefCount(cmdObj); -		if ((wordIdx == 0) && !expand) { -		    /* -		     * We copy the string before trying to find the command by -		     * name. We used to modify the string in place, but this -		     * is not safe because the name resolution handlers could -		     * have side effects that rely on the unmodified string. -		     */ +    TclEmitOpcode(INST_POP, envPtr); +    EnterCmdExtentData(envPtr, cmdIdx, +	    parsePtr->term - parsePtr->commandStart, +	    (envPtr->codeNext-envPtr->codeStart) - startCodeOffset); -		    Tcl_DStringSetLength(&ds, 0); -		    Tcl_DStringAppend(&ds, tokenPtr[1].start,tokenPtr[1].size); - -		    cmdPtr = (Command *) Tcl_FindCommand(interp, -			    Tcl_DStringValue(&ds), -			    (Tcl_Namespace *) cmdNsPtr, /*flags*/ 0); - -		    if ((cmdPtr != NULL) -			    && (cmdPtr->compileProc != NULL) -			    && !(cmdPtr->nsPtr->flags&NS_SUPPRESS_COMPILATION) -			    && !(cmdPtr->flags & CMD_HAS_EXEC_TRACES) -			    && !(iPtr->flags & DONT_COMPILE_CMDS_INLINE)) { -			int savedNumCmds = envPtr->numCommands; -			unsigned savedCodeNext = -				envPtr->codeNext - envPtr->codeStart; -			int update = 0, code; - -			/* -			 * Mark the start of the command; the proper bytecode -			 * length will be updated later. There is no need to -			 * do this for the first bytecode in the compile env, -			 * as the check is done before calling -			 * TclNRExecuteByteCode(). Do emit an INST_START_CMD in -			 * special cases where the first bytecode is in a -			 * loop, to insure that the corresponding command is -			 * counted properly. Compilers for commands able to -			 * produce such a beast (currently 'while 1' only) set -			 * envPtr->atCmdStart to 0 in order to signal this -			 * case. [Bug 1752146] -			 * -			 * Note that the environment is initialised with -			 * atCmdStart=1 to avoid emitting ISC for the first -			 * command. -			 */ - -			if (envPtr->atCmdStart) { -			    if (savedCodeNext != 0) { -				/* -				 * Increase the number of commands being -				 * started at the current point. Note that -				 * this depends on the exact layout of the -				 * INST_START_CMD's operands, so be careful! -				 */ - -				unsigned char *fixPtr = envPtr->codeNext - 4; - -				TclStoreInt4AtPtr(TclGetUInt4AtPtr(fixPtr)+1, -					fixPtr); -			    } -			} else { -			    TclEmitInstInt4(INST_START_CMD, 0, envPtr); -			    TclEmitInt4(1, envPtr); -			    update = 1; -			} - -			code = cmdPtr->compileProc(interp, parsePtr, cmdPtr, -				envPtr); - -			if (code == TCL_OK) { -			    if (update) { -				/* -				 * Fix the bytecode length. -				 */ - -				unsigned char *fixPtr = envPtr->codeStart -					+ savedCodeNext + 1; -				unsigned fixLen = envPtr->codeNext -					- envPtr->codeStart - savedCodeNext; - -				TclStoreInt4AtPtr(fixLen, fixPtr); -			    } -			    goto finishCommand; -			} - -			if (envPtr->atCmdStart && savedCodeNext != 0) { -			    /* -			     * Decrease the number of commands being started -			     * at the current point. Note that this depends on -			     * the exact layout of the INST_START_CMD's -			     * operands, so be careful! -			     */ - -			    unsigned char *fixPtr = envPtr->codeNext - 4; - -			    TclStoreInt4AtPtr(TclGetUInt4AtPtr(fixPtr)-1, -				    fixPtr); -			} - -			/* -			 * Restore numCommands and codeNext to their correct -			 * values, removing any commands compiled before the -			 * failure to produce bytecode got reported. [Bugs -			 * 705406 and 735055] -			 */ - -			envPtr->numCommands = savedNumCmds; -			envPtr->codeNext = envPtr->codeStart + savedCodeNext; -		    } +    /* +     * TIP #280: Free full form of per-word line data and insert the reduced +     * form now +     */ -		    /* -		     * No compile procedure so push the word. If the command -		     * was found, push a CmdName object to reduce runtime -		     * lookups. Mark this as a command name literal to reduce -		     * shimmering.  -		     */ +    envPtr->line = cmdLine; +    envPtr->clNext = clNext; +    ckfree(eclPtr->loc[wlineat].line); +    ckfree(eclPtr->loc[wlineat].next); +    eclPtr->loc[wlineat].line = wlines; +    eclPtr->loc[wlineat].next = NULL; -		    objIndex = TclRegisterNewCmdLiteral(envPtr, -			    tokenPtr[1].start, tokenPtr[1].size); -		    if (cmdPtr != NULL) { -			TclSetCmdNameObj(interp, -				envPtr->literalArrayPtr[objIndex].objPtr, -				cmdPtr); -		    } -		} else { -		    /* -		     * Simple argument word of a command. We reach this if and -		     * only if the command word was not compiled for whatever -		     * reason. Register the literal's location for use by -		     * uplevel, etc. commands, should they encounter it -		     * unmodified. We care only if the we are in a context -		     * which already allows absolute counting. -		     */ +    TclCheckStackDepth(depth, envPtr); +    return cmdIdx; +} -		    objIndex = TclRegisterNewLiteral(envPtr, -			    tokenPtr[1].start, tokenPtr[1].size); +void +TclCompileScript( +    Tcl_Interp *interp,		/* Used for error and status reporting. Also +				 * serves as context for finding and compiling +				 * commands. May not be NULL. */ +    const char *script,		/* The source script to compile. */ +    int numBytes,		/* Number of bytes in script. If < 0, the +				 * script consists of all bytes up to the +				 * first null character. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    int lastCmdIdx = -1;	/* Index into envPtr->cmdMapPtr of the last +				 * command this routine compiles into bytecode. +				 * Initial value of -1 indicates this routine +				 * has not yet generated any bytecode. */ +    const char *p = script;	/* Where we are in our compile. */ +    int depth = TclGetStackDepth(envPtr); + +    if (envPtr->iPtr == NULL) { +	Tcl_Panic("TclCompileScript() called on uninitialized CompileEnv"); +    } -		    if (envPtr->clNext) { -			TclContinuationsEnterDerived( -				envPtr->literalArrayPtr[objIndex].objPtr, -				tokenPtr[1].start - envPtr->source, -				eclPtr->loc[wlineat].next[wordIdx]); -		    } -		} -		TclEmitPush(objIndex, envPtr); -	    } /* for loop */ +    /* Each iteration compiles one command from the script. */ +    while (numBytes > 0) { +	Tcl_Parse parse; +	const char *next; + +	if (TCL_OK != Tcl_ParseCommand(interp, p, numBytes, 0, &parse)) {  	    /* -	     * Emit an invoke instruction for the command. We skip this if a -	     * compile procedure was found for the command. +	     * Compile bytecodes to report the parse error at runtime.  	     */ -	    if (expand) { -		/* -		 * The stack depth during argument expansion can only be -		 * managed at runtime, as the number of elements in the -		 * expanded lists is not known at compile time. We adjust here -		 * the stack depth estimate so that it is correct after the -		 * command with expanded arguments returns. -		 * -		 * The end effect of this command's invocation is that all the -		 * words of the command are popped from the stack, and the -		 * result is pushed: the stack top changes by (1-wordIdx). -		 * -		 * Note that the estimates are not correct while the command -		 * is being prepared and run, INST_EXPAND_STKTOP is not -		 * stack-neutral in general. -		 */ +	    Tcl_LogCommandInfo(interp, script, parse.commandStart, +		    parse.term + 1 - parse.commandStart); +	    TclCompileSyntaxError(interp, envPtr); +	    return; +	} -		TclEmitOpcode(INST_INVOKE_EXPANDED, envPtr); -		TclAdjustStackDepth((1-wordIdx), envPtr); -	    } else if (wordIdx > 0) { -		/* -		 * Save PC -> command map for the TclArgumentBC* functions. -		 */ +#ifdef TCL_COMPILE_DEBUG +	/* +	 * If tracing, print a line for each top level command compiled. +	 * TODO: Suppress when numWords == 0 ? +	 */ -		int isnew; -		Tcl_HashEntry *hePtr = Tcl_CreateHashEntry(&eclPtr->litInfo, -			INT2PTR(envPtr->codeNext - envPtr->codeStart), -			&isnew); +	if ((tclTraceCompile >= 1) && (envPtr->procPtr == NULL)) { +	    int commandLength = parse.term - parse.commandStart; +	    fprintf(stdout, "  Compiling: "); +	    TclPrintSource(stdout, parse.commandStart, +		    TclMin(commandLength, 55)); +	    fprintf(stdout, "\n"); +	} +#endif -		Tcl_SetHashValue(hePtr, INT2PTR(wlineat)); -		if (wordIdx <= 255) { -		    TclEmitInstInt1(INST_INVOKE_STK1, wordIdx, envPtr); -		} else { -		    TclEmitInstInt4(INST_INVOKE_STK4, wordIdx, envPtr); -		} -	    } +	/* +	 * TIP #280: Count newlines before the command start. +	 * (See test info-30.33). +	 */ -	    /* -	     * Update the compilation environment structure and record the -	     * offsets of the source and code for the command. -	     */ +	TclAdvanceLines(&envPtr->line, p, parse.commandStart); +	TclAdvanceContinuations(&envPtr->line, &envPtr->clNext, +		parse.commandStart - envPtr->source); + +	/* +	 * Advance parser to the next command in the script. +	 */ -	finishCommand: -	    EnterCmdExtentData(envPtr, currCmdIndex, commandLength, -		    (envPtr->codeNext-envPtr->codeStart) - startCodeOffset); -	    isFirstCmd = 0; +	next = parse.commandStart + parse.commandSize; +	numBytes -= next - p; +	p = next; +	if (parse.numWords == 0) {  	    /* -	     * TIP #280: Free full form of per-word line data and insert the -	     * reduced form now +	     * The "command" parsed has no words.  In this case we can skip +	     * the rest of the loop body.  With no words, clearly +	     * CompileCommandTokens() has nothing to do.  Since the parser +	     * aggressively sucks up leading comment and white space, +	     * including newlines, parse.commandStart must be pointing at +	     * either the end of script, or a command-terminating semi-colon. +	     * In either case, the TclAdvance*() calls have nothing to do. +	     * Finally, when no words are parsed, no tokens have been +	     * allocated at parse.tokenPtr so there's also nothing for +	     * Tcl_FreeParse() to do. +	     * +	     * The advantage of this shortcut is that CompileCommandTokens() +	     * can be written with an assumption that parse.numWords > 0, with +	     * the implication the CCT() always generates bytecode.  	     */ +	    continue; +	} -	    ckfree(eclPtr->loc[wlineat].line); -	    ckfree(eclPtr->loc[wlineat].next); -	    eclPtr->loc[wlineat].line = wlines; -	    eclPtr->loc[wlineat].next = NULL; -	} /* end if parsePtr->numWords > 0 */ +	lastCmdIdx = CompileCommandTokens(interp, &parse, envPtr);  	/* -	 * Advance to the next command in the script. +	 * TIP #280: Track lines in the just compiled command.  	 */ -	next = parsePtr->commandStart + parsePtr->commandSize; -	bytesLeft -= next - p; -	p = next; +	TclAdvanceLines(&envPtr->line, parse.commandStart, p); +	TclAdvanceContinuations(&envPtr->line, &envPtr->clNext, +		p - envPtr->source); +	Tcl_FreeParse(&parse); +    } +    if (lastCmdIdx == -1) {  	/* -	 * TIP #280: Track lines in the just compiled command. +	 * Compiling the script yielded no bytecode.  The script must be all +	 * whitespace, comments, and empty commands.  Such scripts are defined +	 * to successfully produce the empty string result, so we emit the +	 * simple bytecode that makes that happen.  	 */ -	TclAdvanceLines(&cmdLine, parsePtr->commandStart, p); -	TclAdvanceContinuations(&cmdLine, &clNext, p - envPtr->source); -	Tcl_FreeParse(parsePtr); -    } while (bytesLeft > 0); - -    /* -     * TIP #280: Bring the line counts in the CompEnv up to date. -     *	See tests info-30.33,34,35 . -     */ - -    envPtr->line = cmdLine; -    envPtr->clNext = clNext; - -    /* -     * If the source script yielded no instructions (e.g., if it was empty), -     * push an empty string as the command's result. -     */ +	PushStringLiteral(envPtr, ""); +    } else { +	/* +	 * We compiled at least one command to bytecode.  The routine +	 * CompileCommandTokens() follows the bytecode of each compiled +	 * command with an INST_POP, so that stack balance is maintained when +	 * several commands are in sequence.  (The result of each command is +	 * thrown away before moving on to the next command).  For the last +	 * command compiled, we need to undo that INST_POP so that the result +	 * of the last command becomes the result of the script.  The code +	 * here removes that trailing INST_POP. +	 */ -    if (envPtr->codeNext == entryCodeNext) { -	TclEmitPush(TclRegisterNewLiteral(envPtr, "", 0), envPtr); +	envPtr->cmdMapPtr[lastCmdIdx].numCodeBytes--; +	envPtr->codeNext--; +	envPtr->currStackDepth++;      } - -    envPtr->numSrcBytes = p - script; -    TclStackFree(interp, parsePtr); -    Tcl_DStringFree(&ds); +    TclCheckStackDepth(depth+1, envPtr);  }  /* @@ -1942,7 +2300,7 @@ TclCompileVarSubst(  	localVar = TclFindCompiledLocal(name, nameBytes, localVarName, envPtr);      }      if (localVar < 0) { -	TclEmitPush(TclRegisterNewLiteral(envPtr, name, nameBytes), envPtr); +	PushLiteral(envPtr, name, nameBytes);      }      /* @@ -1954,7 +2312,7 @@ TclCompileVarSubst(      if (tokenPtr->numComponents == 1) {  	if (localVar < 0) { -	    TclEmitOpcode(INST_LOAD_SCALAR_STK, envPtr); +	    TclEmitOpcode(INST_LOAD_STK, envPtr);  	} else if (localVar <= 255) {  	    TclEmitInstInt1(INST_LOAD_SCALAR1, localVar, envPtr);  	} else { @@ -1984,11 +2342,12 @@ TclCompileTokens(      Tcl_DString textBuffer;	/* Holds concatenated chars from adjacent  				 * TCL_TOKEN_TEXT, TCL_TOKEN_BS tokens. */      char buffer[TCL_UTF_MAX]; -    int i, numObjsToConcat, length; +    int i, numObjsToConcat, length, adjust;      unsigned char *entryCodeNext = envPtr->codeNext;  #define NUM_STATIC_POS 20      int isLiteral, maxNumCL, numCL;      int *clPosition = NULL; +    int depth = TclGetStackDepth(envPtr);      /*       * For the handling of continuation lines in literals we first check if @@ -2021,12 +2380,13 @@ TclCompileTokens(  	clPosition = ckalloc(maxNumCL * sizeof(int));      } +    adjust = 0;      Tcl_DStringInit(&textBuffer);      numObjsToConcat = 0;      for ( ;  count > 0;  count--, tokenPtr++) {  	switch (tokenPtr->type) {  	case TCL_TOKEN_TEXT: -	    Tcl_DStringAppend(&textBuffer, tokenPtr->start, tokenPtr->size); +	    TclDStringAppendToken(&textBuffer, tokenPtr);  	    TclAdvanceLines(&envPtr->line, tokenPtr->start,  		    tokenPtr->start + tokenPtr->size);  	    break; @@ -2064,6 +2424,7 @@ TclCompileTokens(  		    clPosition[numCL] = clPos;  		    numCL ++;  		} +		adjust++;  	    }  	    break; @@ -2073,24 +2434,23 @@ TclCompileTokens(  	     */  	    if (Tcl_DStringLength(&textBuffer) > 0) { -		int literal = TclRegisterNewLiteral(envPtr, -			Tcl_DStringValue(&textBuffer), -			Tcl_DStringLength(&textBuffer)); +		int literal = TclRegisterDStringLiteral(envPtr, &textBuffer);  		TclEmitPush(literal, envPtr);  		numObjsToConcat++;  		Tcl_DStringFree(&textBuffer);  		if (numCL) { -		    TclContinuationsEnter( -			    envPtr->literalArrayPtr[literal].objPtr, numCL, -			    clPosition); +		    TclContinuationsEnter(TclFetchLiteral(envPtr, literal), +			    numCL, clPosition);  		}  		numCL = 0;  	    } +	    envPtr->line += adjust;  	    TclCompileScript(interp, tokenPtr->start+1,  		    tokenPtr->size-2, envPtr); +	    envPtr->line -= adjust;  	    numObjsToConcat++;  	    break; @@ -2102,9 +2462,7 @@ TclCompileTokens(  	    if (Tcl_DStringLength(&textBuffer) > 0) {  		int literal; -		literal = TclRegisterNewLiteral(envPtr, -			Tcl_DStringValue(&textBuffer), -			Tcl_DStringLength(&textBuffer)); +		literal = TclRegisterDStringLiteral(envPtr, &textBuffer);  		TclEmitPush(literal, envPtr);  		numObjsToConcat++;  		Tcl_DStringFree(&textBuffer); @@ -2127,15 +2485,12 @@ TclCompileTokens(       */      if (Tcl_DStringLength(&textBuffer) > 0) { -	int literal; +	int literal = TclRegisterDStringLiteral(envPtr, &textBuffer); -	literal = TclRegisterNewLiteral(envPtr, Tcl_DStringValue(&textBuffer), -		Tcl_DStringLength(&textBuffer));  	TclEmitPush(literal, envPtr);  	numObjsToConcat++; -  	if (numCL) { -	    TclContinuationsEnter(envPtr->literalArrayPtr[literal].objPtr, +	    TclContinuationsEnter(TclFetchLiteral(envPtr, literal),  		    numCL, clPosition);  	}  	numCL = 0; @@ -2146,11 +2501,11 @@ TclCompileTokens(       */      while (numObjsToConcat > 255) { -	TclEmitInstInt1(INST_CONCAT1, 255, envPtr); +	TclEmitInstInt1(INST_STR_CONCAT1, 255, envPtr);  	numObjsToConcat -= 254;	/* concat pushes 1 obj, the result */      }      if (numObjsToConcat > 1) { -	TclEmitInstInt1(INST_CONCAT1, numObjsToConcat, envPtr); +	TclEmitInstInt1(INST_STR_CONCAT1, numObjsToConcat, envPtr);      }      /* @@ -2158,7 +2513,7 @@ TclCompileTokens(       */      if (envPtr->codeNext == entryCodeNext) { -	TclEmitPush(TclRegisterNewLiteral(envPtr, "", 0), envPtr); +	PushStringLiteral(envPtr, "");      }      Tcl_DStringFree(&textBuffer); @@ -2170,6 +2525,7 @@ TclCompileTokens(      if (maxNumCL) {  	ckfree(clPosition);      } +    TclCheckStackDepth(depth+1, envPtr);  }  /* @@ -2217,7 +2573,7 @@ TclCompileCmdWord(  	 */  	TclCompileTokens(interp, tokenPtr, count, envPtr); -	TclEmitOpcode(INST_EVAL_STK, envPtr); +	TclEmitInvoke(envPtr, INST_EVAL_STK);      }  } @@ -2273,19 +2629,19 @@ TclCompileExprWords(      wordPtr = tokenPtr;      for (i = 0;  i < numWords;  i++) { -	TclCompileTokens(interp, wordPtr+1, wordPtr->numComponents, envPtr); +	CompileTokens(envPtr, wordPtr, interp);  	if (i < (numWords - 1)) { -	    TclEmitPush(TclRegisterNewLiteral(envPtr, " ", 1), envPtr); +	    PushStringLiteral(envPtr, " ");  	}  	wordPtr += wordPtr->numComponents + 1;      }      concatItems = 2*numWords - 1;      while (concatItems > 255) { -	TclEmitInstInt1(INST_CONCAT1, 255, envPtr); +	TclEmitInstInt1(INST_STR_CONCAT1, 255, envPtr);  	concatItems -= 254;      }      if (concatItems > 1) { -	TclEmitInstInt1(INST_CONCAT1, concatItems, envPtr); +	TclEmitInstInt1(INST_STR_CONCAT1, concatItems, envPtr);      }      TclEmitOpcode(INST_EXPR_STK, envPtr);  } @@ -2319,21 +2675,17 @@ TclCompileNoOp(  {      Tcl_Token *tokenPtr;      int i; -    int savedStackDepth = envPtr->currStackDepth;      tokenPtr = parsePtr->tokenPtr;      for (i = 1; i < parsePtr->numWords; i++) {  	tokenPtr = tokenPtr + tokenPtr->numComponents + 1; -	envPtr->currStackDepth = savedStackDepth;  	if (tokenPtr->type != TCL_TOKEN_SIMPLE_WORD) { -	    TclCompileTokens(interp, tokenPtr+1, tokenPtr->numComponents, -		    envPtr); +	    CompileTokens(envPtr, tokenPtr, interp);  	    TclEmitOpcode(INST_POP, envPtr);  	}      } -    envPtr->currStackDepth = savedStackDepth; -    TclEmitPush(TclRegisterNewLiteral(envPtr, "", 0), envPtr); +    PushStringLiteral(envPtr, "");      return TCL_OK;  } @@ -2382,6 +2734,10 @@ TclInitByteCodeObj(      int i, isNew;      Interp *iPtr; +    if (envPtr->iPtr == NULL) { +	Tcl_Panic("TclInitByteCodeObj() called on uninitialized CompileEnv"); +    } +      iPtr = envPtr->iPtr;      codeBytes = envPtr->codeNext - envPtr->codeStart; @@ -2439,7 +2795,29 @@ TclInitByteCodeObj(      p += TCL_ALIGN(codeBytes);		/* align object array */      codePtr->objArrayPtr = (Tcl_Obj **) p;      for (i = 0;  i < numLitObjects;  i++) { -	codePtr->objArrayPtr[i] = envPtr->literalArrayPtr[i].objPtr; +	Tcl_Obj *fetched = TclFetchLiteral(envPtr, i); + +	if (objPtr == fetched) { +	    /* +	     * Prevent circular reference where the bytecode intrep of +	     * a value contains a literal which is that same value. +	     * If this is allowed to happen, refcount decrements may not +	     * reach zero, and memory may leak.  Bugs 467523, 3357771 +	     * +	     * NOTE:  [Bugs 3392070, 3389764] We make a copy based completely +	     * on the string value, and do not call Tcl_DuplicateObj() so we +             * can be sure we do not have any lingering cycles hiding in +	     * the intrep. +	     */ +	    int numBytes; +	    const char *bytes = Tcl_GetStringFromObj(objPtr, &numBytes); + +	    codePtr->objArrayPtr[i] = Tcl_NewStringObj(bytes, numBytes); +	    Tcl_IncrRefCount(codePtr->objArrayPtr[i]); +	    TclReleaseLiteral((Tcl_Interp *)iPtr, objPtr); +	} else { +	    codePtr->objArrayPtr[i] = fetched; +	}      }      p += TCL_ALIGN(objArrayBytes);	/* align exception range array */ @@ -2464,7 +2842,7 @@ TclInitByteCodeObj(  #else      nextPtr = EncodeCmdLocMap(envPtr, codePtr, (unsigned char *) p);      if (((size_t)(nextPtr - p)) != cmdLocBytes) { -	Tcl_Panic("TclInitByteCodeObj: encoded cmd location bytes %d != expected size %d", (nextPtr - p), cmdLocBytes); +	Tcl_Panic("TclInitByteCodeObj: encoded cmd location bytes %lu != expected size %lu", (unsigned long)(nextPtr - p), (unsigned long)cmdLocBytes);      }  #endif @@ -2487,7 +2865,7 @@ TclInitByteCodeObj(       */      TclFreeIntRep(objPtr); -    objPtr->internalRep.otherValuePtr = codePtr; +    objPtr->internalRep.twoPtrValue.ptr1 = codePtr;      objPtr->typePtr = &tclByteCodeType;      /* @@ -2499,6 +2877,9 @@ TclInitByteCodeObj(  	    &isNew), envPtr->extCmdMapPtr);      envPtr->extCmdMapPtr = NULL; +    /* We've used up the CompileEnv.  Mark as uninitialized. */ +    envPtr->iPtr = NULL; +      codePtr->localCachePtr = NULL;  } @@ -2913,6 +3294,7 @@ TclCreateExceptRange(  				 * new ExceptionRange structure. */  {      register ExceptionRange *rangePtr; +    register ExceptionAux *auxPtr;      int index = envPtr->exceptArrayNext;      if (index >= envPtr->exceptArrayEnd) { @@ -2924,12 +3306,16 @@ TclCreateExceptRange(  	size_t currBytes =  		envPtr->exceptArrayNext * sizeof(ExceptionRange); +	size_t currBytes2 = envPtr->exceptArrayNext * sizeof(ExceptionAux);  	int newElems = 2*envPtr->exceptArrayEnd;  	size_t newBytes = newElems * sizeof(ExceptionRange); +	size_t newBytes2 = newElems * sizeof(ExceptionAux);  	if (envPtr->mallocedExceptArray) {  	    envPtr->exceptArrayPtr =  		    ckrealloc(envPtr->exceptArrayPtr, newBytes); +	    envPtr->exceptAuxArrayPtr = +		    ckrealloc(envPtr->exceptAuxArrayPtr, newBytes2);  	} else {  	    /*  	     * envPtr->exceptArrayPtr isn't a ckalloc'd pointer, so we must @@ -2937,9 +3323,12 @@ TclCreateExceptRange(  	     */  	    ExceptionRange *newPtr = ckalloc(newBytes); +	    ExceptionAux *newPtr2 = ckalloc(newBytes2);  	    memcpy(newPtr, envPtr->exceptArrayPtr, currBytes); +	    memcpy(newPtr2, envPtr->exceptAuxArrayPtr, currBytes2);  	    envPtr->exceptArrayPtr = newPtr; +	    envPtr->exceptAuxArrayPtr = newPtr2;  	    envPtr->mallocedExceptArray = 1;  	}  	envPtr->exceptArrayEnd = newElems; @@ -2954,10 +3343,294 @@ TclCreateExceptRange(      rangePtr->breakOffset = -1;      rangePtr->continueOffset = -1;      rangePtr->catchOffset = -1; +    auxPtr = &envPtr->exceptAuxArrayPtr[index]; +    auxPtr->supportsContinue = 1; +    auxPtr->stackDepth = envPtr->currStackDepth; +    auxPtr->expandTarget = envPtr->expandCount; +    auxPtr->expandTargetDepth = -1; +    auxPtr->numBreakTargets = 0; +    auxPtr->breakTargets = NULL; +    auxPtr->allocBreakTargets = 0; +    auxPtr->numContinueTargets = 0; +    auxPtr->continueTargets = NULL; +    auxPtr->allocContinueTargets = 0;      return index;  }  /* + * --------------------------------------------------------------------- + * + * TclGetInnermostExceptionRange -- + * + *	Returns the innermost exception range that covers the current code + *	creation point, and (optionally) the stack depth that is expected at + *	that point. Relies on the fact that the range has a numCodeBytes = -1 + *	when it is being populated and that inner ranges come after outer + *	ranges. + * + * --------------------------------------------------------------------- + */ + +ExceptionRange * +TclGetInnermostExceptionRange( +    CompileEnv *envPtr, +    int returnCode, +    ExceptionAux **auxPtrPtr) +{ +    int exnIdx = -1, i; + +    for (i=0 ; i<envPtr->exceptArrayNext ; i++) { +	ExceptionRange *rangePtr = &envPtr->exceptArrayPtr[i]; + +	if (CurrentOffset(envPtr) >= rangePtr->codeOffset && +		(rangePtr->numCodeBytes == -1 || CurrentOffset(envPtr) < +			rangePtr->codeOffset+rangePtr->numCodeBytes) && +		(returnCode != TCL_CONTINUE || +			envPtr->exceptAuxArrayPtr[i].supportsContinue)) { +	    exnIdx = i; +	} +    } +    if (exnIdx == -1) { +	return NULL; +    } +    if (auxPtrPtr) { +	*auxPtrPtr = &envPtr->exceptAuxArrayPtr[exnIdx]; +    } +    return &envPtr->exceptArrayPtr[exnIdx]; +} + +/* + * --------------------------------------------------------------------- + * + * TclAddLoopBreakFixup, TclAddLoopContinueFixup -- + * + *	Adds a place that wants to break/continue to the loop exception range + *	tracking that will be fixed up once the loop can be finalized. These + *	functions will generate an INST_JUMP4 that will be fixed up during the + *	loop finalization. + * + * --------------------------------------------------------------------- + */ + +void +TclAddLoopBreakFixup( +    CompileEnv *envPtr, +    ExceptionAux *auxPtr) +{ +    int range = auxPtr - envPtr->exceptAuxArrayPtr; + +    if (envPtr->exceptArrayPtr[range].type != LOOP_EXCEPTION_RANGE) { +	Tcl_Panic("trying to add 'break' fixup to full exception range"); +    } + +    if (++auxPtr->numBreakTargets > auxPtr->allocBreakTargets) { +	auxPtr->allocBreakTargets *= 2; +	auxPtr->allocBreakTargets += 2; +	if (auxPtr->breakTargets) { +	    auxPtr->breakTargets = ckrealloc(auxPtr->breakTargets, +		    sizeof(int) * auxPtr->allocBreakTargets); +	} else { +	    auxPtr->breakTargets = +		    ckalloc(sizeof(int) * auxPtr->allocBreakTargets); +	} +    } +    auxPtr->breakTargets[auxPtr->numBreakTargets - 1] = CurrentOffset(envPtr); +    TclEmitInstInt4(INST_JUMP4, 0, envPtr); +} + +void +TclAddLoopContinueFixup( +    CompileEnv *envPtr, +    ExceptionAux *auxPtr) +{ +    int range = auxPtr - envPtr->exceptAuxArrayPtr; + +    if (envPtr->exceptArrayPtr[range].type != LOOP_EXCEPTION_RANGE) { +	Tcl_Panic("trying to add 'continue' fixup to full exception range"); +    } + +    if (++auxPtr->numContinueTargets > auxPtr->allocContinueTargets) { +	auxPtr->allocContinueTargets *= 2; +	auxPtr->allocContinueTargets += 2; +	if (auxPtr->continueTargets) { +	    auxPtr->continueTargets = ckrealloc(auxPtr->continueTargets, +		    sizeof(int) * auxPtr->allocContinueTargets); +	} else { +	    auxPtr->continueTargets = +		    ckalloc(sizeof(int) * auxPtr->allocContinueTargets); +	} +    } +    auxPtr->continueTargets[auxPtr->numContinueTargets - 1] = +	    CurrentOffset(envPtr); +    TclEmitInstInt4(INST_JUMP4, 0, envPtr); +} + +/* + * --------------------------------------------------------------------- + * + * TclCleanupStackForBreakContinue -- + * + *	Ditch the extra elements from the auxiliary stack and the main stack. + *	How to do this exactly depends on whether there are any elements on + *	the auxiliary stack to pop. + * + * --------------------------------------------------------------------- + */ + +void +TclCleanupStackForBreakContinue( +    CompileEnv *envPtr, +    ExceptionAux *auxPtr) +{ +    int savedStackDepth = envPtr->currStackDepth; +    int toPop = envPtr->expandCount - auxPtr->expandTarget; + +    if (toPop > 0) { +	while (toPop --> 0) { +	    TclEmitOpcode(INST_EXPAND_DROP, envPtr); +	} +	TclAdjustStackDepth(auxPtr->expandTargetDepth - envPtr->currStackDepth, +		envPtr); +	envPtr->currStackDepth = auxPtr->expandTargetDepth; +    } +    toPop = envPtr->currStackDepth - auxPtr->stackDepth; +    while (toPop --> 0) { +	TclEmitOpcode(INST_POP, envPtr); +    } +    envPtr->currStackDepth = savedStackDepth; +} + +/* + * --------------------------------------------------------------------- + * + * StartExpanding -- + * + *	Pushes an INST_EXPAND_START and does some additional housekeeping so + *	that the [break] and [continue] compilers can use an exception-free + *	issue to discard it. + * + * --------------------------------------------------------------------- + */ + +static void +StartExpanding( +    CompileEnv *envPtr) +{ +    int i; + +    TclEmitOpcode(INST_EXPAND_START, envPtr); + +    /* +     * Update inner exception ranges with information about the environment +     * where this expansion started. +     */ + +    for (i=0 ; i<envPtr->exceptArrayNext ; i++) { +	ExceptionRange *rangePtr = &envPtr->exceptArrayPtr[i]; +	ExceptionAux *auxPtr = &envPtr->exceptAuxArrayPtr[i]; + +	/* +	 * Ignore loops unless they're still being built. +	 */ + +	if (rangePtr->codeOffset > CurrentOffset(envPtr)) { +	    continue; +	} +	if (rangePtr->numCodeBytes != -1) { +	    continue; +	} + +	/* +	 * Adequate condition: further out loops and further in exceptions +	 * don't actually need this information. +	 */ + +	if (auxPtr->expandTarget == envPtr->expandCount) { +	    auxPtr->expandTargetDepth = envPtr->currStackDepth; +	} +    } + +    /* +     * There's now one more expansion being processed on the auxiliary stack. +     */ + +    envPtr->expandCount++; +} + +/* + * --------------------------------------------------------------------- + * + * TclFinalizeLoopExceptionRange -- + * + *	Finalizes a loop exception range, binding the registered [break] and + *	[continue] implementations so that they jump to the correct place. + *	Note that this must only be called after *all* the exception range + *	target offsets have been set. + * + * --------------------------------------------------------------------- + */ + +void +TclFinalizeLoopExceptionRange( +    CompileEnv *envPtr, +    int range) +{ +    ExceptionRange *rangePtr = &envPtr->exceptArrayPtr[range]; +    ExceptionAux *auxPtr = &envPtr->exceptAuxArrayPtr[range]; +    int i, offset; +    unsigned char *site; + +    if (rangePtr->type != LOOP_EXCEPTION_RANGE) { +	Tcl_Panic("trying to finalize a loop exception range"); +    } + +    /* +     * Do the jump fixups. Note that these are always issued as INST_JUMP4 so +     * there is no need to fuss around with updating code offsets. +     */ + +    for (i=0 ; i<auxPtr->numBreakTargets ; i++) { +	site = envPtr->codeStart + auxPtr->breakTargets[i]; +	offset = rangePtr->breakOffset - auxPtr->breakTargets[i]; +	TclUpdateInstInt4AtPc(INST_JUMP4, offset, site); +    } +    for (i=0 ; i<auxPtr->numContinueTargets ; i++) { +	site = envPtr->codeStart + auxPtr->continueTargets[i]; +	if (rangePtr->continueOffset == -1) { +	    int j; + +	    /* +	     * WTF? Can't bind, so revert to an INST_CONTINUE. Not enough +	     * space to do anything else. +	     */ + +	    *site = INST_CONTINUE; +	    for (j=0 ; j<4 ; j++) { +		*++site = INST_NOP; +	    } +	} else { +	    offset = rangePtr->continueOffset - auxPtr->continueTargets[i]; +	    TclUpdateInstInt4AtPc(INST_JUMP4, offset, site); +	} +    } + +    /* +     * Drop the arrays we were holding the only reference to. +     */ + +    if (auxPtr->breakTargets) { +	ckfree(auxPtr->breakTargets); +	auxPtr->breakTargets = NULL; +	auxPtr->numBreakTargets = 0; +    } +    if (auxPtr->continueTargets) { +	ckfree(auxPtr->continueTargets); +	auxPtr->continueTargets = NULL; +	auxPtr->numContinueTargets = 0; +    } +} + +/*   *----------------------------------------------------------------------   *   * TclCreateAuxData -- @@ -3316,12 +3989,221 @@ TclFixupForwardJump(  		    rangePtr->type);  	}      } + +    for (k = 0 ; k < envPtr->exceptArrayNext ; k++) { +	ExceptionAux *auxPtr = &envPtr->exceptAuxArrayPtr[k]; +	int i; + +	for (i=0 ; i<auxPtr->numBreakTargets ; i++) { +	    if (jumpFixupPtr->codeOffset < auxPtr->breakTargets[i]) { +		auxPtr->breakTargets[i] += 3; +	    } +	} +	for (i=0 ; i<auxPtr->numContinueTargets ; i++) { +	    if (jumpFixupPtr->codeOffset < auxPtr->continueTargets[i]) { +		auxPtr->continueTargets[i] += 3; +	    } +	} +    } +      return 1;			/* the jump was grown */  }  /*   *----------------------------------------------------------------------   * + * TclEmitInvoke -- + * + *	Emit one of the invoke-related instructions, wrapping it if necessary + *	in code that ensures that any break or continue operation passing + *	through it gets the stack unwinding correct, converting it into an + *	internal jump if in an appropriate context. + * + * Results: + *	None + * + * Side effects: + *	Issues the jump with all correct stack management. May create another + *	loop exception range; pointers to ExceptionRange and ExceptionAux + *	structures should not be held across this call. + * + *---------------------------------------------------------------------- + */ + +void +TclEmitInvoke( +    CompileEnv *envPtr, +    int opcode, +    ...) +{ +    va_list argList; +    ExceptionRange *rangePtr; +    ExceptionAux *auxBreakPtr, *auxContinuePtr; +    int arg1, arg2, wordCount = 0, expandCount = 0; +    int loopRange = 0, breakRange = 0, continueRange = 0; +    int cleanup, depth = TclGetStackDepth(envPtr); +     +    /* +     * Parse the arguments. +     */ + +    va_start(argList, opcode); +    switch (opcode) { +    case INST_INVOKE_STK1: +	wordCount = arg1 = cleanup = va_arg(argList, int); +	arg2 = 0; +	break; +    case INST_INVOKE_STK4: +	wordCount = arg1 = cleanup = va_arg(argList, int); +	arg2 = 0; +	break; +    case INST_INVOKE_REPLACE: +	arg1 = va_arg(argList, int); +	arg2 = va_arg(argList, int); +	wordCount = arg1 + arg2 - 1; +	cleanup = arg1 + 1; +	break; +    default: +	Tcl_Panic("unexpected opcode"); +    case INST_EVAL_STK: +	wordCount = cleanup = 1; +	arg1 = arg2 = 0; +	break; +    case INST_RETURN_STK: +	wordCount = cleanup = 2; +	arg1 = arg2 = 0; +	break; +    case INST_INVOKE_EXPANDED: +	wordCount = arg1 = cleanup = va_arg(argList, int); +	arg2 = 0; +	expandCount = 1; +	break; +    } +    va_end(argList); + +    /* +     * Determine if we need to handle break and continue exceptions with a +     * special handling exception range (so that we can correctly unwind the +     * stack). +     * +     * These must be done separately; they can be different (especially for +     * calls from inside a [for] increment clause). +     */ + +    rangePtr = TclGetInnermostExceptionRange(envPtr, TCL_BREAK, &auxBreakPtr); +    if (rangePtr == NULL || rangePtr->type != LOOP_EXCEPTION_RANGE) { +	auxBreakPtr = NULL; +    } else if (auxBreakPtr->stackDepth == envPtr->currStackDepth-wordCount +	    && auxBreakPtr->expandTarget == envPtr->expandCount-expandCount) { +	auxBreakPtr = NULL; +    } else { +	breakRange = auxBreakPtr - envPtr->exceptAuxArrayPtr; +    } + +    rangePtr = TclGetInnermostExceptionRange(envPtr, TCL_CONTINUE, +	    &auxContinuePtr); +    if (rangePtr == NULL || rangePtr->type != LOOP_EXCEPTION_RANGE) { +	auxContinuePtr = NULL; +    } else if (auxContinuePtr->stackDepth == envPtr->currStackDepth-wordCount +	    && auxContinuePtr->expandTarget == envPtr->expandCount-expandCount) { +	auxContinuePtr = NULL; +    } else { +	continueRange = auxBreakPtr - envPtr->exceptAuxArrayPtr; +    } + +    if (auxBreakPtr != NULL || auxContinuePtr != NULL) { +	loopRange = TclCreateExceptRange(LOOP_EXCEPTION_RANGE, envPtr); +	ExceptionRangeStarts(envPtr, loopRange); +    } + +    /* +     * Issue the invoke itself. +     */ + +    switch (opcode) { +    case INST_INVOKE_STK1: +	TclEmitInstInt1(INST_INVOKE_STK1, arg1, envPtr); +	break; +    case INST_INVOKE_STK4: +	TclEmitInstInt4(INST_INVOKE_STK4, arg1, envPtr); +	break; +    case INST_INVOKE_EXPANDED: +	TclEmitOpcode(INST_INVOKE_EXPANDED, envPtr); +	envPtr->expandCount--; +	TclAdjustStackDepth(1 - arg1, envPtr); +	break; +    case INST_EVAL_STK: +	TclEmitOpcode(INST_EVAL_STK, envPtr); +	break; +    case INST_RETURN_STK: +	TclEmitOpcode(INST_RETURN_STK, envPtr); +	break; +    case INST_INVOKE_REPLACE: +	TclEmitInstInt4(INST_INVOKE_REPLACE, arg1, envPtr); +	TclEmitInt1(arg2, envPtr); +	TclAdjustStackDepth(-1, envPtr); /* Correction to stack depth calcs */ +	break; +    } + +    /* +     * If we're generating a special wrapper exception range, we need to +     * finish that up now. +     */ + +    if (auxBreakPtr != NULL || auxContinuePtr != NULL) { +	int savedStackDepth = envPtr->currStackDepth; +	int savedExpandCount = envPtr->expandCount; +	JumpFixup nonTrapFixup; + +	if (auxBreakPtr != NULL) { +	    auxBreakPtr = envPtr->exceptAuxArrayPtr + breakRange; +	} +	if (auxContinuePtr != NULL) { +	    auxContinuePtr = envPtr->exceptAuxArrayPtr + continueRange; +	} + +	ExceptionRangeEnds(envPtr, loopRange); +	TclEmitForwardJump(envPtr, TCL_UNCONDITIONAL_JUMP, &nonTrapFixup); + +	/* +	 * Careful! When generating these stack unwinding sequences, the depth +	 * of stack in the cases where they are taken is not the same as if +	 * the exception is not taken. +	 */ + +	if (auxBreakPtr != NULL) { +	    TclAdjustStackDepth(-1, envPtr); + +	    ExceptionRangeTarget(envPtr, loopRange, breakOffset); +	    TclCleanupStackForBreakContinue(envPtr, auxBreakPtr); +	    TclAddLoopBreakFixup(envPtr, auxBreakPtr); +	    TclAdjustStackDepth(1, envPtr); + +	    envPtr->currStackDepth = savedStackDepth; +	    envPtr->expandCount = savedExpandCount; +	} + +	if (auxContinuePtr != NULL) { +	    TclAdjustStackDepth(-1, envPtr); + +	    ExceptionRangeTarget(envPtr, loopRange, continueOffset); +	    TclCleanupStackForBreakContinue(envPtr, auxContinuePtr); +	    TclAddLoopContinueFixup(envPtr, auxContinuePtr); +	    TclAdjustStackDepth(1, envPtr); + +	    envPtr->currStackDepth = savedStackDepth; +	    envPtr->expandCount = savedExpandCount; +	} + +	TclFinalizeLoopExceptionRange(envPtr, loopRange); +	TclFixupForwardJumpToHere(envPtr, &nonTrapFixup, 127); +    } +    TclCheckStackDepth(depth+1-cleanup, envPtr); +} + +/* + *---------------------------------------------------------------------- + *   * TclGetInstructionTable --   *   *	Returns a pointer to the table describing Tcl bytecode instructions. @@ -3347,7 +4229,7 @@ TclGetInstructionTable(void)  /*   *--------------------------------------------------------------   * - * TclRegisterAuxDataType -- + * RegisterAuxDataType --   *   *	This procedure is called to register a new AuxData type in the table   *	of all AuxData types supported by Tcl. @@ -3363,8 +4245,8 @@ TclGetInstructionTable(void)   *--------------------------------------------------------------   */ -void -TclRegisterAuxDataType( +static void +RegisterAuxDataType(      const AuxDataType *typePtr)	/* Information about object type; storage must  				 * be statically allocated (must live forever;  				 * will not be deallocated). */ @@ -3465,11 +4347,12 @@ TclInitAuxDataTypeTable(void)      Tcl_InitHashTable(&auxDataTypeTable, TCL_STRING_KEYS);      /* -     * There are only two AuxData type at this time, so register them here. +     * There are only three AuxData types at this time, so register them here.       */ -    TclRegisterAuxDataType(&tclForeachInfoType); -    TclRegisterAuxDataType(&tclJumptableInfoType); +    RegisterAuxDataType(&tclForeachInfoType); +    RegisterAuxDataType(&tclJumptableInfoType); +    RegisterAuxDataType(&tclDictUpdateInfoType);  }  /* @@ -3841,7 +4724,7 @@ Tcl_Obj *  TclDisassembleByteCodeObj(      Tcl_Obj *objPtr)		/* The bytecode object to disassemble. */  { -    ByteCode *codePtr = objPtr->internalRep.otherValuePtr; +    ByteCode *codePtr = objPtr->internalRep.twoPtrValue.ptr1;      unsigned char *codeStart, *codeLimit, *pc;      unsigned char *codeDeltaNext, *codeLengthNext;      unsigned char *srcDeltaNext, *srcLengthNext; @@ -4214,6 +5097,11 @@ FormatInstruction(  	    }  	    Tcl_AppendPrintfToObj(bufferObj, "%%v%u ", (unsigned) opnd);  	    break; +	case OPERAND_SCLS1: +	    opnd = TclGetUInt1AtPtr(pc+numBytes); numBytes++; +	    Tcl_AppendPrintfToObj(bufferObj, "%s ", +		    tclStringClassTable[opnd].name); +	    break;  	case OPERAND_NONE:  	default:  	    break; @@ -4346,7 +5234,11 @@ TclGetInnerContext(          if (!objPtr) {              Tcl_Panic("InnerContext: bad tos -- appending null object");          } -        if (objPtr->refCount<=0 || objPtr->refCount==0x61616161) { +        if ((objPtr->refCount<=0) +#ifdef TCL_MEM_DEBUG +                || (objPtr->refCount==0x61616161) +#endif +        ) {              Tcl_Panic("InnerContext: bad tos -- appending freed object %p",                      objPtr);          } @@ -4366,7 +5258,7 @@ TclGetInnerContext(   *----------------------------------------------------------------------   */ -MODULE_SCOPE Tcl_Obj * +Tcl_Obj *  TclNewInstNameObj(      unsigned char inst)  { @@ -4426,7 +5318,7 @@ PrintSourceToObj(      int maxChars)		/* Maximum number of chars to print. */  {      register const char *p; -    register int i = 0; +    register int i = 0, len;      if (stringPtr == NULL) {  	Tcl_AppendToObj(appendObj, "\"\"", -1); @@ -4435,32 +5327,50 @@ PrintSourceToObj(      Tcl_AppendToObj(appendObj, "\"", -1);      p = stringPtr; -    for (;  (*p != '\0') && (i < maxChars);  p++, i++) { -	switch (*p) { +    for (;  (*p != '\0') && (i < maxChars);  p+=len) { +	Tcl_UniChar ch; + +	len = TclUtfToUniChar(p, &ch); +	switch (ch) {  	case '"':  	    Tcl_AppendToObj(appendObj, "\\\"", -1); +	    i += 2;  	    continue;  	case '\f':  	    Tcl_AppendToObj(appendObj, "\\f", -1); +	    i += 2;  	    continue;  	case '\n':  	    Tcl_AppendToObj(appendObj, "\\n", -1); +	    i += 2;  	    continue;  	case '\r':  	    Tcl_AppendToObj(appendObj, "\\r", -1); +	    i += 2;  	    continue;  	case '\t':  	    Tcl_AppendToObj(appendObj, "\\t", -1); +	    i += 2;  	    continue;  	case '\v':  	    Tcl_AppendToObj(appendObj, "\\v", -1); +	    i += 2;  	    continue;  	default: -	    Tcl_AppendPrintfToObj(appendObj, "%c", *p); +	    if (ch < 0x20 || ch >= 0x7f) { +		Tcl_AppendPrintfToObj(appendObj, "\\u%04x", ch); +		i += 6; +	    } else { +		Tcl_AppendPrintfToObj(appendObj, "%c", ch); +		i++; +	    }  	    continue;  	}      }      Tcl_AppendToObj(appendObj, "\"", -1); +    if (*p != '\0') { +	Tcl_AppendToObj(appendObj, "...", -1); +    }  }  #ifdef TCL_COMPILE_STATS @@ -4490,7 +5400,13 @@ RecordByteCodeStats(  				 * to add to accumulated statistics. */  {      Interp *iPtr = (Interp *) *codePtr->interpHandle; -    register ByteCodeStats *statsPtr = &iPtr->stats; +    register ByteCodeStats *statsPtr; + +    if (iPtr == NULL) { +	/* Avoid segfaulting in case we're called in a deleted interp */ +	return; +    } +    statsPtr = &(iPtr->stats);      statsPtr->numCompilations++;      statsPtr->totalSrcBytes += (double) codePtr->numSrcBytes; @@ -4518,6 +5434,5 @@ RecordByteCodeStats(   * c-basic-offset: 4   * fill-column: 78   * tab-width: 8 - * indent-tabs-mode: nil   * End:   */ | 
