diff options
Diffstat (limited to 'generic/tclVar.c')
| -rw-r--r-- | generic/tclVar.c | 8638 | 
1 files changed, 4882 insertions, 3756 deletions
| diff --git a/generic/tclVar.c b/generic/tclVar.c index 7a8741a..4694cd8 100644 --- a/generic/tclVar.c +++ b/generic/tclVar.c @@ -1,8 +1,8 @@ -/*  +/*   * tclVar.c --   * - *	This file contains routines that implement Tcl variables - *	(both scalars and arrays). + *	This file contains routines that implement Tcl variables (both scalars + *	and arrays).   *   *	The implementation of arrays is modelled after an initial   *	implementation by Mark Diekhans and Karl Lehenbauer. @@ -10,81 +10,357 @@   * Copyright (c) 1987-1994 The Regents of the University of California.   * Copyright (c) 1994-1997 Sun Microsystems, Inc.   * Copyright (c) 1998-1999 by Scriptics Corporation. - * Copyright (c) 2001 by Kevin B. Kenny.  All rights reserved. - * - * See the file "license.terms" for information on usage and redistribution - * of this file, and for a DISCLAIMER OF ALL WARRANTIES. + * Copyright (c) 2001 by Kevin B. Kenny. All rights reserved. + * Copyright (c) 2007 Miguel Sofer   * - * RCS: @(#) $Id: tclVar.c,v 1.49 2002/02/27 06:38:39 hobbs Exp $ + * See the file "license.terms" for information on usage and redistribution of + * this file, and for a DISCLAIMER OF ALL WARRANTIES.   */  #include "tclInt.h" -#include "tclPort.h" +#include "tclOOInt.h" + +/* + * Prototypes for the variable hash key methods. + */ + +static Tcl_HashEntry *	AllocVarEntry(Tcl_HashTable *tablePtr, void *keyPtr); +static void		FreeVarEntry(Tcl_HashEntry *hPtr); +static int		CompareVarKeys(void *keyPtr, Tcl_HashEntry *hPtr); + +static const Tcl_HashKeyType tclVarHashKeyType = { +    TCL_HASH_KEY_TYPE_VERSION,	/* version */ +    0,				/* flags */ +    TclHashObjKey,		/* hashKeyProc */ +    CompareVarKeys,		/* compareKeysProc */ +    AllocVarEntry,		/* allocEntryProc */ +    FreeVarEntry		/* freeEntryProc */ +}; + +static inline Var *	VarHashCreateVar(TclVarHashTable *tablePtr, +			    Tcl_Obj *key, int *newPtr); +static inline Var *	VarHashFirstVar(TclVarHashTable *tablePtr, +			    Tcl_HashSearch *searchPtr); +static inline Var *	VarHashNextVar(Tcl_HashSearch *searchPtr); +static inline void	CleanupVar(Var *varPtr, Var *arrayPtr); + +#define VarHashGetValue(hPtr) \ +    ((Var *) ((char *)hPtr - TclOffset(VarInHash, entry))) + +/* + * NOTE: VarHashCreateVar increments the recount of its key argument. + * All callers that will call Tcl_DecrRefCount on that argument must + * call Tcl_IncrRefCount on it before passing it in.  This requirement + * can bubble up to callers of callers .... etc. + */ + +static inline Var * +VarHashCreateVar( +    TclVarHashTable *tablePtr, +    Tcl_Obj *key, +    int *newPtr) +{ +    Tcl_HashEntry *hPtr = Tcl_CreateHashEntry(&tablePtr->table, +	    key, newPtr); + +    if (hPtr) { +	return VarHashGetValue(hPtr); +    } else { +	return NULL; +    } +} + +#define VarHashFindVar(tablePtr, key) \ +    VarHashCreateVar((tablePtr), (key), NULL) + +#define VarHashInvalidateEntry(varPtr) \ +    ((varPtr)->flags |= VAR_DEAD_HASH) + +#define VarHashDeleteEntry(varPtr) \ +    Tcl_DeleteHashEntry(&(((VarInHash *) varPtr)->entry)) + +#define VarHashFirstEntry(tablePtr, searchPtr) \ +    Tcl_FirstHashEntry(&(tablePtr)->table, (searchPtr)) + +#define VarHashNextEntry(searchPtr) \ +    Tcl_NextHashEntry((searchPtr)) + +static inline Var * +VarHashFirstVar( +    TclVarHashTable *tablePtr, +    Tcl_HashSearch *searchPtr) +{ +    Tcl_HashEntry *hPtr = VarHashFirstEntry(tablePtr, searchPtr); + +    if (hPtr) { +	return VarHashGetValue(hPtr); +    } else { +	return NULL; +    } +} + +static inline Var * +VarHashNextVar( +    Tcl_HashSearch *searchPtr) +{ +    Tcl_HashEntry *hPtr = VarHashNextEntry(searchPtr); + +    if (hPtr) { +	return VarHashGetValue(hPtr); +    } else { +	return NULL; +    } +} + +#define VarHashGetKey(varPtr) \ +    (((VarInHash *)(varPtr))->entry.key.objPtr) + +#define VarHashDeleteTable(tablePtr) \ +    Tcl_DeleteHashTable(&(tablePtr)->table) + +/* + * The strings below are used to indicate what went wrong when a variable + * access is denied. + */ + +static const char *noSuchVar =		"no such variable"; +static const char *isArray =		"variable is array"; +static const char *needArray =		"variable isn't array"; +static const char *noSuchElement =	"no such element in array"; +static const char *danglingElement = +	"upvar refers to element in deleted array"; +static const char *danglingVar = +	"upvar refers to variable in deleted namespace"; +static const char *badNamespace =	"parent namespace doesn't exist"; +static const char *missingName =	"missing variable name"; +static const char *isArrayElement = +	"name refers to an element in an array";  /* - * The strings below are used to indicate what went wrong when a - * variable access is denied. + * A test to see if we are in a call frame that has local variables. This is + * true if we are inside a procedure body.   */ -static char *noSuchVar =	"no such variable"; -static char *isArray =		"variable is array"; -static char *needArray =	"variable isn't array"; -static char *noSuchElement =	"no such element in array"; -static char *danglingElement =	"upvar refers to element in deleted array"; -static char *danglingVar =     "upvar refers to variable in deleted namespace"; -static char *badNamespace =	"parent namespace doesn't exist"; -static char *missingName =	"missing variable name"; -static char *isArrayElement =	"name refers to an element in an array"; +#define HasLocalVars(framePtr) ((framePtr)->isProcCallFrame & FRAME_IS_PROC)  /* - * Forward references to procedures defined later in this file: + * The following structure describes an enumerative search in progress on an + * array variable; this are invoked with options to the "array" command.   */ -static  char *		CallTraces _ANSI_ARGS_((Interp *iPtr, Var *arrayPtr, -			    Var *varPtr, char *part1, char *part2, -			    int flags, int *resultTypePtr)); -static void		CleanupVar _ANSI_ARGS_((Var *varPtr, -			    Var *arrayPtr)); -static void		DeleteSearches _ANSI_ARGS_((Var *arrayVarPtr)); -static void		DeleteArray _ANSI_ARGS_((Interp *iPtr, -			    char *arrayName, Var *varPtr, int flags)); -static int		MakeUpvar _ANSI_ARGS_(( -			    Interp *iPtr, CallFrame *framePtr, -			    char *otherP1, char *otherP2, int otherFlags, -			    CONST char *myName, int myFlags)); -static Var *		NewVar _ANSI_ARGS_((void)); -static ArraySearch *	ParseSearchId _ANSI_ARGS_((Tcl_Interp *interp, -			    Var *varPtr, char *varName, Tcl_Obj *handleObj)); -static void		VarErrMsg _ANSI_ARGS_((Tcl_Interp *interp, -			    char *part1, char *part2, char *operation, -			    char *reason)); -static int		SetArraySearchObj _ANSI_ARGS_((Tcl_Interp *interp, -			    Tcl_Obj *objPtr)); +typedef struct ArraySearch { +    int id;			/* Integer id used to distinguish among +				 * multiple concurrent searches for the same +				 * array. */ +    struct Var *varPtr;		/* Pointer to array variable that's being +				 * searched. */ +    Tcl_HashSearch search;	/* Info kept by the hash module about progress +				 * through the array. */ +    Tcl_HashEntry *nextEntry;	/* Non-null means this is the next element to +				 * be enumerated (it's leftover from the +				 * Tcl_FirstHashEntry call or from an "array +				 * anymore" command). NULL means must call +				 * Tcl_NextHashEntry to get value to +				 * return. */ +    struct ArraySearch *nextPtr;/* Next in list of all active searches for +				 * this variable, or NULL if this is the last +				 * one. */ +} ArraySearch; + +/* + * Forward references to functions defined later in this file: + */ + +static void		AppendLocals(Tcl_Interp *interp, Tcl_Obj *listPtr, +			    Tcl_Obj *patternPtr, int includeLinks); +static void		DeleteSearches(Interp *iPtr, Var *arrayVarPtr); +static void		DeleteArray(Interp *iPtr, Tcl_Obj *arrayNamePtr, +			    Var *varPtr, int flags, int index); +static Tcl_Var		ObjFindNamespaceVar(Tcl_Interp *interp, +			    Tcl_Obj *namePtr, Tcl_Namespace *contextNsPtr, +			    int flags); +static int		ObjMakeUpvar(Tcl_Interp *interp, +			    CallFrame *framePtr, Tcl_Obj *otherP1Ptr, +			    const char *otherP2, const int otherFlags, +			    Tcl_Obj *myNamePtr, int myFlags, int index); +static ArraySearch *	ParseSearchId(Tcl_Interp *interp, const Var *varPtr, +			    Tcl_Obj *varNamePtr, Tcl_Obj *handleObj); +static void		UnsetVarStruct(Var *varPtr, Var *arrayPtr, +			    Interp *iPtr, Tcl_Obj *part1Ptr, +			    Tcl_Obj *part2Ptr, int flags, int index); +static int		SetArraySearchObj(Tcl_Interp *interp, +			    Tcl_Obj *objPtr); + +/* + * Functions defined in this file that may be exported in the future for use + * by the bytecode compiler and engine or to the public interface. + */ + +MODULE_SCOPE Var *	TclLookupSimpleVar(Tcl_Interp *interp, +			    Tcl_Obj *varNamePtr, int flags, const int create, +			    const char **errMsgPtr, int *indexPtr); + +static Tcl_DupInternalRepProc	DupLocalVarName; +static Tcl_FreeInternalRepProc	FreeLocalVarName; +static Tcl_UpdateStringProc	PanicOnUpdateVarName; + +static Tcl_FreeInternalRepProc	FreeParsedVarName; +static Tcl_DupInternalRepProc	DupParsedVarName; +static Tcl_UpdateStringProc	UpdateParsedVarName; + +static Tcl_UpdateStringProc	PanicOnUpdateVarName; +static Tcl_SetFromAnyProc	PanicOnSetVarName; + +/* + * Types of Tcl_Objs used to cache variable lookups. + * + * localVarName - INTERNALREP DEFINITION: + *   ptrAndLongRep.ptr:   pointer to name obj in varFramePtr->localCache + *			  or NULL if it is this same obj + *   ptrAndLongRep.value: index into locals table + * + * nsVarName - INTERNALREP DEFINITION: + *   twoPtrValue.ptr1:	pointer to the namespace containing the reference + *   twoPtrValue.ptr2:	pointer to the corresponding Var + * + * parsedVarName - INTERNALREP DEFINITION: + *   twoPtrValue.ptr1:	pointer to the array name Tcl_Obj, or NULL if it is a + *			scalar variable + *   twoPtrValue.ptr2:	pointer to the element name string (owned by this + *			Tcl_Obj), or NULL if it is a scalar variable + */ + +static const Tcl_ObjType localVarNameType = { +    "localVarName", +    FreeLocalVarName, DupLocalVarName, PanicOnUpdateVarName, PanicOnSetVarName +}; + +/* + * Caching of namespace variables disabled: no simple way was found to avoid + * interfering with the resolver's idea of variable existence. A cached + * varName may keep a variable's name in the namespace's hash table, which is + * the resolver's criterion for existence (see test namespace-17.10). + */ + +#define ENABLE_NS_VARNAME_CACHING 0 + +#if ENABLE_NS_VARNAME_CACHING +static Tcl_FreeInternalRepProc FreeNsVarName; +static Tcl_DupInternalRepProc DupNsVarName; + +static const Tcl_ObjType tclNsVarNameType = { +    "namespaceVarName", +    FreeNsVarName, DupNsVarName, PanicOnUpdateVarName, PanicOnSetVarName +}; +#endif + +static const Tcl_ObjType tclParsedVarNameType = { +    "parsedVarName", +    FreeParsedVarName, DupParsedVarName, UpdateParsedVarName, PanicOnSetVarName +};  /*   * Type of Tcl_Objs used to speed up array searches.   *   * INTERNALREP DEFINITION: - *   twoPtrValue.ptr1 = searchIdNumber as offset from (char*)NULL - *   twoPtrValue.ptr2 = variableNameStartInString as offset from (char*)NULL + *   twoPtrValue.ptr1:	searchIdNumber (cast to pointer) + *   twoPtrValue.ptr2:	variableNameStartInString (cast to pointer)   * - * Note that the value stored in ptr2 is the offset into the string of - * the start of the variable name and not the address of the variable - * name itself, as this can be safely copied. + * Note that the value stored in ptr2 is the offset into the string of the + * start of the variable name and not the address of the variable name itself, + * as this can be safely copied.   */ -Tcl_ObjType tclArraySearchType = { + +const Tcl_ObjType tclArraySearchType = {      "array search",      NULL, NULL, NULL, SetArraySearchObj  }; + +Var * +TclVarHashCreateVar( +    TclVarHashTable *tablePtr, +    const char *key, +    int *newPtr) +{ +    Tcl_Obj *keyPtr; +    Var *varPtr; + +    keyPtr = Tcl_NewStringObj(key, -1); +    Tcl_IncrRefCount(keyPtr); +    varPtr = VarHashCreateVar(tablePtr, keyPtr, newPtr); +    Tcl_DecrRefCount(keyPtr); + +    return varPtr; +} + +/* + *---------------------------------------------------------------------- + * + * TclCleanupVar -- + * + *	This function is called when it looks like it may be OK to free up a + *	variable's storage. If the variable is in a hashtable, its Var + *	structure and hash table entry will be freed along with those of its + *	containing array, if any. This function is called, for example, when + *	a trace on a variable deletes a variable. + * + * Results: + *	None. + * + * Side effects: + *	If the variable (or its containing array) really is dead and in a + *	hashtable, then its Var structure, and possibly its hash table entry, + *	is freed up. + * + *---------------------------------------------------------------------- + */ + +static inline void +CleanupVar( +    Var *varPtr,		/* Pointer to variable that may be a candidate +				 * for being expunged. */ +    Var *arrayPtr)		/* Array that contains the variable, or NULL +				 * if this variable isn't an array element. */ +{ +    if (TclIsVarUndefined(varPtr) && TclIsVarInHash(varPtr) +	    && !TclIsVarTraced(varPtr) +	    && (VarHashRefCount(varPtr) == !TclIsVarDeadHash(varPtr))) { +	if (VarHashRefCount(varPtr) == 0) { +	    ckfree(varPtr); +	} else { +	    VarHashDeleteEntry(varPtr); +	} +    } +    if (arrayPtr != NULL && TclIsVarUndefined(arrayPtr) && +	    TclIsVarInHash(arrayPtr) && !TclIsVarTraced(arrayPtr) && +	    (VarHashRefCount(arrayPtr) == !TclIsVarDeadHash(arrayPtr))) { +	if (VarHashRefCount(arrayPtr) == 0) { +	    ckfree(arrayPtr); +	} else { +	    VarHashDeleteEntry(arrayPtr); +	} +    } +} +void +TclCleanupVar( +    Var *varPtr,		/* Pointer to variable that may be a candidate +				 * for being expunged. */ +    Var *arrayPtr)		/* Array that contains the variable, or NULL +				 * if this variable isn't an array element. */ +{ +    CleanupVar(varPtr, arrayPtr); +}  /*   *----------------------------------------------------------------------   *   * TclLookupVar --   * - *	This procedure is used by virtually all of the variable code to - *	locate a variable given its name(s). + *	This function is used to locate a variable given its name(s). It has + *	been mostly superseded by TclObjLookupVar, it is now only used by the + *	trace code. It is kept in tcl8.5 mainly because it is in the internal + *	stubs table, so that some extension may be calling it.   *   * Results:   *	The return value is a pointer to the variable structure indicated by @@ -98,14 +374,14 @@ Tcl_ObjType tclArraySearchType = {   *   *	If the variable isn't found and creation wasn't specified, or some   *	other error occurs, NULL is returned and an error message is left in - *	the interp's result if TCL_LEAVE_ERR_MSG is set in flags.  + *	the interp's result if TCL_LEAVE_ERR_MSG is set in flags.   * - *	Note: it's possible for the variable returned to be VAR_UNDEFINED - *	even if createPart1 or createPart2 are 1 (these only cause the hash - *	table entry or array to be created). For example, the variable might - *	be a global that has been unset but is still referenced by a - *	procedure, or a variable that has been unset but it only being kept - *	in existence (if VAR_UNDEFINED) by a trace. + *	Note: it's possible for the variable returned to be VAR_UNDEFINED even + *	if createPart1 or createPart2 are 1 (these only cause the hash table + *	entry or array to be created). For example, the variable might be a + *	global that has been unset but is still referenced by a procedure, or + *	a variable that has been unset but it only being kept in existence (if + *	VAR_UNDEFINED) by a trace.   *   * Side effects:   *	New hashtable entries may be created if createPart1 or createPart2 @@ -115,340 +391,837 @@ Tcl_ObjType tclArraySearchType = {   */  Var * -TclLookupVar(interp, part1, part2, flags, msg, createPart1, createPart2, -        arrayPtrPtr) -    Tcl_Interp *interp;		/* Interpreter to use for lookup. */ -    register char *part1;	/* If part2 isn't NULL, this is the name of -				 * an array. Otherwise, this -				 * is a full variable name that could -				 * include a parenthesized array element. */ -    char *part2;		/* Name of element within array, or NULL. */ -    int flags;			/* Only TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +TclLookupVar( +    Tcl_Interp *interp,		/* Interpreter to use for lookup. */ +    const char *part1,		/* If part2 isn't NULL, this is the name of an +				 * array. Otherwise, this is a full variable +				 * name that could include a parenthesized +				 * array element. */ +    const char *part2,		/* Name of element within array, or NULL. */ +    int flags,			/* Only TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * and TCL_LEAVE_ERR_MSG bits matter. */ +    const char *msg,		/* Verb to use in error messages, e.g. "read" +				 * or "set". Only needed if TCL_LEAVE_ERR_MSG +				 * is set in flags. */ +    int createPart1,		/* If 1, create hash table entry for part 1 of +				 * name, if it doesn't already exist. If 0, +				 * return error if it doesn't exist. */ +    int createPart2,		/* If 1, create hash table entry for part 2 of +				 * name, if it doesn't already exist. If 0, +				 * return error if it doesn't exist. */ +    Var **arrayPtrPtr)		/* If the name refers to an element of an +				 * array, *arrayPtrPtr gets filled in with +				 * address of array variable. Otherwise this +				 * is set to NULL. */ +{ +    Var *varPtr; +    Tcl_Obj *part1Ptr = Tcl_NewStringObj(part1, -1); + +    if (createPart1) { +	Tcl_IncrRefCount(part1Ptr); +    } + +    varPtr = TclObjLookupVar(interp, part1Ptr, part2, flags, msg, +	    createPart1, createPart2, arrayPtrPtr); + +    TclDecrRefCount(part1Ptr); +    return varPtr; +} + +/* + *---------------------------------------------------------------------- + * + * TclObjLookupVar, TclObjLookupVarEx -- + * + *	This function is used by virtually all of the variable code to locate + *	a variable given its name(s). The parsing into array/element + *	components and (if possible) the lookup results are cached in + *	part1Ptr, which is converted to one of the varNameTypes. + * + * Results: + *	The return value is a pointer to the variable structure indicated by + *	part1Ptr and part2, or NULL if the variable couldn't be found. If * + *	the variable is found, *arrayPtrPtr is filled with the address of the + *	variable structure for the array that contains the variable (or NULL + *	if the variable is a scalar). If the variable can't be found and + *	either createPart1 or createPart2 are 1, a new as-yet-undefined + *	(VAR_UNDEFINED) variable structure is created, entered into a hash + *	table, and returned. + * + *	If the variable isn't found and creation wasn't specified, or some + *	other error occurs, NULL is returned and an error message is left in + *	the interp's result if TCL_LEAVE_ERR_MSG is set in flags. + * + *	Note: it's possible for the variable returned to be VAR_UNDEFINED even + *	if createPart1 or createPart2 are 1 (these only cause the hash table + *	entry or array to be created). For example, the variable might be a + *	global that has been unset but is still referenced by a procedure, or + *	a variable that has been unset but it only being kept in existence (if + *	VAR_UNDEFINED) by a trace. + * + * Side effects: + *	New hashtable entries may be created if createPart1 or createPart2 + *	are 1. The object part1Ptr is converted to one of localVarNameType, + *	tclNsVarNameType or tclParsedVarNameType and caches as much of the + *	lookup as it can. + *	When createPart1 is 1, callers must IncrRefCount part1Ptr if they + *	plan to DecrRefCount it. + * + *---------------------------------------------------------------------- + */ + +Var * +TclObjLookupVar( +    Tcl_Interp *interp,		/* Interpreter to use for lookup. */ +    register Tcl_Obj *part1Ptr,	/* If part2 isn't NULL, this is the name of an +				 * array. Otherwise, this is a full variable +				 * name that could include a parenthesized +				 * array element. */ +    const char *part2,		/* Name of element within array, or NULL. */ +    int flags,			/* Only TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * and TCL_LEAVE_ERR_MSG bits matter. */ +    const char *msg,		/* Verb to use in error messages, e.g. "read" +				 * or "set". Only needed if TCL_LEAVE_ERR_MSG +				 * is set in flags. */ +    const int createPart1,	/* If 1, create hash table entry for part 1 of +				 * name, if it doesn't already exist. If 0, +				 * return error if it doesn't exist. */ +    const int createPart2,	/* If 1, create hash table entry for part 2 of +				 * name, if it doesn't already exist. If 0, +				 * return error if it doesn't exist. */ +    Var **arrayPtrPtr)		/* If the name refers to an element of an +				 * array, *arrayPtrPtr gets filled in with +				 * address of array variable. Otherwise this +				 * is set to NULL. */ +{ +    Tcl_Obj *part2Ptr = NULL; +    Var *resPtr; + +    if (part2) { +	part2Ptr = Tcl_NewStringObj(part2, -1); +	if (createPart2) { +	    Tcl_IncrRefCount(part2Ptr); +	} +    } + +    resPtr = TclObjLookupVarEx(interp, part1Ptr, part2Ptr, +	    flags, msg, createPart1, createPart2, arrayPtrPtr); + +    if (part2Ptr) { +	Tcl_DecrRefCount(part2Ptr); +    } + +    return resPtr; +} + +/* + *	When createPart1 is 1, callers must IncrRefCount part1Ptr if they + *	plan to DecrRefCount it. + *	When createPart2 is 1, callers must IncrRefCount part2Ptr if they + *	plan to DecrRefCount it. + */ +Var * +TclObjLookupVarEx( +    Tcl_Interp *interp,		/* Interpreter to use for lookup. */ +    Tcl_Obj *part1Ptr,		/* If part2Ptr isn't NULL, this is the name of +				 * an array. Otherwise, this is a full +				 * variable name that could include a +				 * parenthesized array element. */ +    Tcl_Obj *part2Ptr,		/* Name of element within array, or NULL. */ +    int flags,			/* Only TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY,  				 * and TCL_LEAVE_ERR_MSG bits matter. */ -    char *msg;			/* Verb to use in error messages, e.g. -				 * "read" or "set". Only needed if -				 * TCL_LEAVE_ERR_MSG is set in flags. */ -    int createPart1;		/* If 1, create hash table entry for part 1 -				 * of name, if it doesn't already exist. If -				 * 0, return error if it doesn't exist. */ -    int createPart2;		/* If 1, create hash table entry for part 2 -				 * of name, if it doesn't already exist. If -				 * 0, return error if it doesn't exist. */ -    Var **arrayPtrPtr;		/* If the name refers to an element of an +    const char *msg,		/* Verb to use in error messages, e.g. "read" +				 * or "set". Only needed if TCL_LEAVE_ERR_MSG +				 * is set in flags. */ +    const int createPart1,	/* If 1, create hash table entry for part 1 of +				 * name, if it doesn't already exist. If 0, +				 * return error if it doesn't exist. */ +    const int createPart2,	/* If 1, create hash table entry for part 2 of +				 * name, if it doesn't already exist. If 0, +				 * return error if it doesn't exist. */ +    Var **arrayPtrPtr)		/* If the name refers to an element of an  				 * array, *arrayPtrPtr gets filled in with -				 * address of array variable. Otherwise -				 * this is set to NULL. */ +				 * address of array variable. Otherwise this +				 * is set to NULL. */  {      Interp *iPtr = (Interp *) interp; +    register Var *varPtr;	/* Points to the variable's in-frame Var +				 * structure. */ +    const char *part1; +    int index, len1, len2; +    int parsed = 0; +    Tcl_Obj *objPtr; +    const Tcl_ObjType *typePtr = part1Ptr->typePtr; +    const char *errMsg = NULL;      CallFrame *varFramePtr = iPtr->varFramePtr; -				/* Points to the procedure call frame whose -				 * variables are currently in use. Same as -				 * the current procedure's frame, if any, -				 * unless an "uplevel" is executing. */ -    Tcl_HashTable *tablePtr;	/* Points to the hashtable, if any, in which -				 * to look up the variable. */ -    Tcl_Var var;                /* Used to search for global names. */ -    Var *varPtr;		/* Points to the Var structure returned for -    				 * the variable. */ -    char *elName;		/* Name of array element or NULL; may be -				 * same as part2, or may be openParen+1. */ -    char *openParen, *closeParen; -                                /* If this procedure parses a name into -				 * array and index, these point to the -				 * parens around the index.  Otherwise they -				 * are NULL. These are needed to restore -				 * the parens after parsing the name. */ -    Namespace *varNsPtr, *cxtNsPtr, *dummy1Ptr, *dummy2Ptr; -    ResolverScheme *resPtr; -    Tcl_HashEntry *hPtr; -    register char *p; -    int new, i, result; +#if ENABLE_NS_VARNAME_CACHING +    Namespace *nsPtr; +#endif +    const char *part2 = part2Ptr? TclGetString(part2Ptr):NULL; +    char *newPart2 = NULL; -    varPtr = NULL;      *arrayPtrPtr = NULL; -    openParen = closeParen = NULL; -    varNsPtr = NULL;		/* set non-NULL if a nonlocal variable */ + +#if ENABLE_NS_VARNAME_CACHING +    if (varFramePtr) { +	nsPtr = varFramePtr->nsPtr; +    } else { +	/* +	 * Some variables in the global ns have to be initialized before the +	 * root call frame is in place. +	 */ + +	nsPtr = NULL; +    } +#endif + +    if (typePtr == &localVarNameType) { +	int localIndex; + +    localVarNameTypeHandling: +	localIndex = (int) part1Ptr->internalRep.ptrAndLongRep.value; +	if (HasLocalVars(varFramePtr) +		&& !(flags & (TCL_GLOBAL_ONLY | TCL_NAMESPACE_ONLY)) +		&& (localIndex < varFramePtr->numCompiledLocals)) { +	    /* +	     * Use the cached index if the names coincide. +	     */ + +	    Tcl_Obj *namePtr = part1Ptr->internalRep.ptrAndLongRep.ptr; +	    Tcl_Obj *checkNamePtr = localName(iPtr->varFramePtr, localIndex); + +	    if ((!namePtr && (checkNamePtr == part1Ptr)) || +		    (namePtr && (checkNamePtr == namePtr))) { +		varPtr = (Var *) &(varFramePtr->compiledLocals[localIndex]); +		goto donePart1; +	    } +	} +	goto doneParsing; +#if ENABLE_NS_VARNAME_CACHING +    } else if (typePtr == &tclNsVarNameType) { +	int useGlobal, useReference; +	Namespace *cachedNsPtr = part1Ptr->internalRep.twoPtrValue.ptr1; +	varPtr = part1Ptr->internalRep.twoPtrValue.ptr2; + +	useGlobal = (cachedNsPtr == iPtr->globalNsPtr) && ( +		(flags & TCL_GLOBAL_ONLY) || +		(part1[0]==':' && part1[1]==':') || +		(!HasLocalVars(varFramePtr) && (nsPtr==iPtr->globalNsPtr))); + +	useReference = useGlobal || ((cachedNsPtr == nsPtr) && ( +		(flags & TCL_NAMESPACE_ONLY) || +		(!HasLocalVars(varFramePtr) && !(flags & TCL_GLOBAL_ONLY) && +		/* +		 * Careful: an undefined ns variable could be hiding a valid +		 * global reference. +		 */ +		!TclIsVarUndefined(varPtr)))); + +	if (useReference && !TclIsVarDeadHash(varPtr)) { +	    /* +	     * A straight global or namespace reference, use it. It isn't so +	     * simple to deal with 'implicit' namespace references, i.e., +	     * those where the reference could be to either a namespace or a +	     * global variable. Those we lookup again. +	     * +	     * If TclIsVarDeadHash(varPtr), this might be a reference to a +	     * variable in a deleted namespace, kept alive by e.g. part1Ptr. +	     * We could conceivably be so unlucky that a new namespace was +	     * created at the same address as the deleted one, so to be safe +	     * we test for a valid hPtr. +	     */ + +	    goto donePart1; +	} +	goto doneParsing; +#endif +    }      /* -     * Parse part1 into array name and index. -     * Always check if part1 is an array element name and allow it only if -     * part2 is not given.    -     * (if one does not care about creating array elements that can't be used -     *  from tcl, and prefer slightly better performance, one can put -     *  the following in an   if (part2 == NULL) { ... } block and remove -     *  the part2's test and error reporting  or move that code in array set) +     * If part1Ptr is a tclParsedVarNameType, separate it into the pre-parsed +     * parts.       */ -    elName = part2; -    for (p = part1; *p ; p++) { -	if (*p == '(') { -	    openParen = p; -	    do { -		p++; -	    } while (*p != '\0'); -	    p--; -	    if (*p == ')') { -		if (part2 != NULL) { -		    openParen = NULL; +    if (typePtr == &tclParsedVarNameType) { +	if (part1Ptr->internalRep.twoPtrValue.ptr1 != NULL) { +	    if (part2Ptr != NULL) { +		/* +		 * ERROR: part1Ptr is already an array element, cannot specify +		 * a part2. +		 */ + +		if (flags & TCL_LEAVE_ERR_MSG) { +		    TclObjVarErrMsg(interp, part1Ptr, part2Ptr, msg, +			    noSuchVar, -1); +		    Tcl_SetErrorCode(interp, "TCL", "VALUE", "VARNAME", NULL); +		} +		return NULL; +	    } +	    part2 = newPart2 = part1Ptr->internalRep.twoPtrValue.ptr2; +	    if (newPart2) { +		part2Ptr = Tcl_NewStringObj(newPart2, -1); +		if (createPart2) { +		    Tcl_IncrRefCount(part2Ptr); +		} +	    } +	    part1Ptr = part1Ptr->internalRep.twoPtrValue.ptr1; +	    typePtr = part1Ptr->typePtr; +	    if (typePtr == &localVarNameType) { +		goto localVarNameTypeHandling; +	    } +	} +	parsed = 1; +    } +    part1 = TclGetStringFromObj(part1Ptr, &len1); + +    if (!parsed && (*(part1 + len1 - 1) == ')')) { +	/* +	 * part1Ptr is possibly an unparsed array element. +	 */ + +	register int i; + +	len2 = -1; +	for (i = 0; i < len1; i++) { +	    if (*(part1 + i) == '(') { +		if (part2Ptr != NULL) {  		    if (flags & TCL_LEAVE_ERR_MSG) { -			VarErrMsg(interp, part1, part2, msg, needArray); +			TclObjVarErrMsg(interp, part1Ptr, part2Ptr, msg, +				needArray, -1); +			Tcl_SetErrorCode(interp, "TCL", "VALUE", "VARNAME", +				NULL);  		    } -		    goto done; +		    return NULL;  		} -		closeParen = p; -		*openParen = 0; -		*closeParen = 0; -		elName = openParen+1; -	    } else { -		openParen = NULL; + +		/* +		 * part1Ptr points to an array element; first copy the element +		 * name to a new string part2. +		 */ + +		part2 = part1 + i + 1; +		len2 = len1 - i - 2; +		len1 = i; + +		newPart2 = ckalloc(len2 + 1); +		memcpy(newPart2, part2, (unsigned) len2); +		*(newPart2+len2) = '\0'; +		part2 = newPart2; +		part2Ptr = Tcl_NewStringObj(newPart2, -1); +		if (createPart2) { +		    Tcl_IncrRefCount(part2Ptr); +		} + +		/* +		 * Free the internal rep of the original part1Ptr, now renamed +		 * objPtr, and set it to tclParsedVarNameType. +		 */ + +		objPtr = part1Ptr; +		TclFreeIntRep(objPtr); +		objPtr->typePtr = &tclParsedVarNameType; + +		/* +		 * Define a new string object to hold the new part1Ptr, i.e., +		 * the array name. Set the internal rep of objPtr, reset +		 * typePtr and part1 to contain the references to the array +		 * name. +		 */ + +		TclNewStringObj(part1Ptr, part1, len1); +		Tcl_IncrRefCount(part1Ptr); + +		objPtr->internalRep.twoPtrValue.ptr1 = part1Ptr; +		objPtr->internalRep.twoPtrValue.ptr2 = (void *) part2; + +		typePtr = part1Ptr->typePtr; +		part1 = TclGetString(part1Ptr); +		break;  	    } -	    break;  	}      } +  doneParsing: +    /* +     * part1Ptr is not an array element; look it up, and convert it to one of +     * the cached types if possible. +     */ + +    TclFreeIntRep(part1Ptr); + +    varPtr = TclLookupSimpleVar(interp, part1Ptr, flags, createPart1, +	    &errMsg, &index); +    if (varPtr == NULL) { +	if ((errMsg != NULL) && (flags & TCL_LEAVE_ERR_MSG)) { +	    TclObjVarErrMsg(interp, part1Ptr, part2Ptr, msg, errMsg, -1); +	    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", +		    TclGetString(part1Ptr), NULL); +	} +	if (newPart2) { +	    Tcl_DecrRefCount(part2Ptr); +	} +	return NULL; +    } +      /* -     * If this namespace has a variable resolver, then give it first -     * crack at the variable resolution.  It may return a Tcl_Var -     * value, it may signal to continue onward, or it may signal -     * an error. +     * Cache the newly found variable if possible.       */ -    if ((flags & TCL_GLOBAL_ONLY) != 0 || iPtr->varFramePtr == NULL) { -        cxtNsPtr = iPtr->globalNsPtr; + +    if (index >= 0) { +	/* +	 * An indexed local variable. +	 */ + +	part1Ptr->typePtr = &localVarNameType; +	if (part1Ptr != localName(iPtr->varFramePtr, index)) { +	    part1Ptr->internalRep.ptrAndLongRep.ptr = +		    localName(iPtr->varFramePtr, index); +	    Tcl_IncrRefCount((Tcl_Obj *) +		    part1Ptr->internalRep.ptrAndLongRep.ptr); +	} else { +	    part1Ptr->internalRep.ptrAndLongRep.ptr = NULL; +	} +	part1Ptr->internalRep.ptrAndLongRep.value = (long) index; +#if ENABLE_NS_VARNAME_CACHING +    } else if (index > -3) { +	/* +	 * A cacheable namespace or global variable. +	 */ + +	Namespace *nsPtr; + +	nsPtr = ((index == -1) ? iPtr->globalNsPtr : varFramePtr->nsPtr); +	varPtr->refCount++; +	part1Ptr->typePtr = &tclNsVarNameType; +	part1Ptr->internalRep.twoPtrValue.ptr1 = nsPtr; +	part1Ptr->internalRep.twoPtrValue.ptr2 = varPtr; +#endif      } else { -        cxtNsPtr = iPtr->varFramePtr->nsPtr; +	/* +	 * At least mark part1Ptr as already parsed. +	 */ + +	part1Ptr->typePtr = &tclParsedVarNameType; +	part1Ptr->internalRep.twoPtrValue.ptr1 = NULL; +	part1Ptr->internalRep.twoPtrValue.ptr2 = NULL;      } -    if (cxtNsPtr->varResProc != NULL || iPtr->resolverPtr != NULL) { -        resPtr = iPtr->resolverPtr; +  donePart1: +#if 0 /* ENABLE_NS_VARNAME_CACHING perhaps? */ +    if (varPtr == NULL) { +	if (flags & TCL_LEAVE_ERR_MSG) { +	    part1 = TclGetString(part1Ptr); +	    TclObjVarErrMsg(interp, part1Ptr, part2Ptr, msg, +		    "cached variable reference is NULL.", -1); +	    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", +		    TclGetString(part1Ptr), NULL); +	} +	return NULL; +    } +#endif +    while (TclIsVarLink(varPtr)) { +	varPtr = varPtr->value.linkPtr; +    } -        if (cxtNsPtr->varResProc) { -            result = (*cxtNsPtr->varResProc)(interp, part1, +    if (part2Ptr != NULL) { +	/* +	 * Array element sought: look it up. +	 */ + +	*arrayPtrPtr = varPtr; +	varPtr = TclLookupArrayElement(interp, part1Ptr, part2Ptr, flags, msg, +		createPart1, createPart2, varPtr, -1); +	if (newPart2) { +	    Tcl_DecrRefCount(part2Ptr); +	} +    } +    return varPtr; +} + +/* + * This flag bit should not interfere with TCL_GLOBAL_ONLY, + * TCL_NAMESPACE_ONLY, or TCL_LEAVE_ERR_MSG; it signals that the variable + * lookup is performed for upvar (or similar) purposes, with slightly + * different rules: + *    - Bug #696893 - variable is either proc-local or in the current + *	namespace; never follow the second (global) resolution path + *    - Bug #631741 - do not use special namespace or interp resolvers + * + * It should also not collide with the (deprecated) TCL_PARSE_PART1 flag + * (Bug #835020) + */ + +#define AVOID_RESOLVERS 0x40000 + +/* + *---------------------------------------------------------------------- + * + * TclLookupSimpleVar -- + * + *	This function is used by to locate a simple variable (i.e., not an + *	array element) given its name. + * + * Results: + *	The return value is a pointer to the variable structure indicated by + *	varName, or NULL if the variable couldn't be found. If the variable + *	can't be found and create is 1, a new as-yet-undefined (VAR_UNDEFINED) + *	variable structure is created, entered into a hash table, and + *	returned. + * + *	If the current CallFrame corresponds to a proc and the variable found + *	is one of the compiledLocals, its index is placed in *indexPtr. + *	Otherwise, *indexPtr will be set to (according to the needs of + *	TclObjLookupVar): + *	    -1 a global reference + *	    -2 a reference to a namespace variable + *	    -3 a non-cachable reference, i.e., one of: + *		. non-indexed local var + *		. a reference of unknown origin; + *		. resolution by a namespace or interp resolver + * + *	If the variable isn't found and creation wasn't specified, or some + *	other error occurs, NULL is returned and the corresponding error + *	message is left in *errMsgPtr. + * + *	Note: it's possible for the variable returned to be VAR_UNDEFINED even + *	if create is 1 (this only causes the hash table entry to be created). + *	For example, the variable might be a global that has been unset but is + *	still referenced by a procedure, or a variable that has been unset but + *	it only being kept in existence (if VAR_UNDEFINED) by a trace. + * + * Side effects: + *	A new hashtable entry may be created if create is 1. + *	Callers must Incr varNamePtr if they plan to Decr it if create is 1. + * + *---------------------------------------------------------------------- + */ + +Var * +TclLookupSimpleVar( +    Tcl_Interp *interp,		/* Interpreter to use for lookup. */ +    Tcl_Obj *varNamePtr,	/* This is a simple variable name that could +				 * represent a scalar or an array. */ +    int flags,			/* Only TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * AVOID_RESOLVERS and TCL_LEAVE_ERR_MSG bits +				 * matter. */ +    const int create,		/* If 1, create hash table entry for varname, +				 * if it doesn't already exist. If 0, return +				 * error if it doesn't exist. */ +    const char **errMsgPtr, +    int *indexPtr) +{ +    Interp *iPtr = (Interp *) interp; +    CallFrame *varFramePtr = iPtr->varFramePtr; +				/* Points to the procedure call frame whose +				 * variables are currently in use. Same as the +				 * current procedure's frame, if any, unless +				 * an "uplevel" is executing. */ +    TclVarHashTable *tablePtr;	/* Points to the hashtable, if any, in which +				 * to look up the variable. */ +    Tcl_Var var;		/* Used to search for global names. */ +    Var *varPtr;		/* Points to the Var structure returned for +				 * the variable. */ +    Namespace *varNsPtr, *cxtNsPtr, *dummy1Ptr, *dummy2Ptr; +    ResolverScheme *resPtr; +    int isNew, i, result, varLen; +    const char *varName = TclGetStringFromObj(varNamePtr, &varLen); + +    varPtr = NULL; +    varNsPtr = NULL;		/* Set non-NULL if a nonlocal variable. */ +    *indexPtr = -3; + +    if (flags & TCL_GLOBAL_ONLY) { +	cxtNsPtr = iPtr->globalNsPtr; +    } else { +	cxtNsPtr = iPtr->varFramePtr->nsPtr; +    } + +    /* +     * If this namespace has a variable resolver, then give it first crack at +     * the variable resolution. It may return a Tcl_Var value, it may signal +     * to continue onward, or it may signal an error. +     */ + +    if ((cxtNsPtr->varResProc != NULL || iPtr->resolverPtr != NULL) +	    && !(flags & AVOID_RESOLVERS)) { +	resPtr = iPtr->resolverPtr; +	if (cxtNsPtr->varResProc) { +	    result = cxtNsPtr->varResProc(interp, varName,  		    (Tcl_Namespace *) cxtNsPtr, flags, &var); -        } else { -            result = TCL_CONTINUE; -        } +	} else { +	    result = TCL_CONTINUE; +	} -        while (result == TCL_CONTINUE && resPtr) { -            if (resPtr->varResProc) { -                result = (*resPtr->varResProc)(interp, part1, +	while (result == TCL_CONTINUE && resPtr) { +	    if (resPtr->varResProc) { +		result = resPtr->varResProc(interp, varName,  			(Tcl_Namespace *) cxtNsPtr, flags, &var); -            } -            resPtr = resPtr->nextPtr; -        } +	    } +	    resPtr = resPtr->nextPtr; +	} -        if (result == TCL_OK) { -            varPtr = (Var *) var; -            goto lookupVarPart2; -        } else if (result != TCL_CONTINUE) { -	    varPtr = (Var *) NULL; -	    /* can't just return here as input string is in an -	     * inconsistent state... */ -	    goto done; -        } +	if (result == TCL_OK) { +	    return (Var *) var; +	} else if (result != TCL_CONTINUE) { +	    return NULL; +	}      }      /* -     * Look up part1. Look it up as either a namespace variable or as a -     * local variable in a procedure call frame (varFramePtr). -     * Interpret part1 as a namespace variable if: +     * Look up varName. Look it up as either a namespace variable or as a +     * local variable in a procedure call frame (varFramePtr). Interpret +     * varName as a namespace variable if:       *    1) so requested by a TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY flag,       *    2) there is no active frame (we're at the global :: scope), -     *    3) the active frame was pushed to define the namespace context -     *       for a "namespace eval" or "namespace inscope" command, +     *    3) the active frame was pushed to define the namespace context for a +     *	     "namespace eval" or "namespace inscope" command,       *    4) the name has namespace qualifiers ("::"s). -     * Otherwise, if part1 is a local variable, search first in the -     * frame's array of compiler-allocated local variables, then in its -     * hashtable for runtime-created local variables. +     * Otherwise, if varName is a local variable, search first in the frame's +     * array of compiler-allocated local variables, then in its hashtable for +     * runtime-created local variables.       * -     * If createPart1 and the variable isn't found, create the variable and, -     * if necessary, create varFramePtr's local var hashtable. +     * If create and the variable isn't found, create the variable and, if +     * necessary, create varFramePtr's local var hashtable.       */      if (((flags & (TCL_GLOBAL_ONLY | TCL_NAMESPACE_ONLY)) != 0) -	    || (varFramePtr == NULL) -	    || !varFramePtr->isProcCallFrame -	    || (strstr(part1, "::") != NULL)) { -	CONST char *tail; -	 +	    || !HasLocalVars(varFramePtr) +	    || (strstr(varName, "::") != NULL)) { +	const char *tail; +	int lookGlobal = (flags & TCL_GLOBAL_ONLY) +		|| (cxtNsPtr == iPtr->globalNsPtr) +		|| ((*varName == ':') && (*(varName+1) == ':')); + +	if (lookGlobal) { +	    *indexPtr = -1; +	    flags = (flags | TCL_GLOBAL_ONLY) & ~TCL_NAMESPACE_ONLY; +	} else { +	    if (flags & AVOID_RESOLVERS) { +		flags = (flags | TCL_NAMESPACE_ONLY); +	    } +	    if (flags & TCL_NAMESPACE_ONLY) { +		*indexPtr = -2; +	    } +	} +  	/* -	 * Don't pass TCL_LEAVE_ERR_MSG, we may yet create the variable, -	 * or otherwise generate our own error! +	 * Don't pass TCL_LEAVE_ERR_MSG, we may yet create the variable, or +	 * otherwise generate our own error!  	 */ -	var = Tcl_FindNamespaceVar(interp, part1, (Tcl_Namespace *) NULL, -		flags & ~TCL_LEAVE_ERR_MSG); -	if (var != (Tcl_Var) NULL) { -            varPtr = (Var *) var; -        } + +	varPtr = (Var *) ObjFindNamespaceVar(interp, varNamePtr, +		(Tcl_Namespace *) cxtNsPtr, +		(flags | AVOID_RESOLVERS) & ~TCL_LEAVE_ERR_MSG);  	if (varPtr == NULL) { -	    if (createPart1) {   /* var wasn't found so create it  */ -		TclGetNamespaceForQualName(interp, part1, (Namespace *) NULL, -			flags, &varNsPtr, &dummy1Ptr, &dummy2Ptr, &tail); +	    Tcl_Obj *tailPtr; +	    if (create) {	/* Var wasn't found so create it. */ +		TclGetNamespaceForQualName(interp, varName, cxtNsPtr, +			flags, &varNsPtr, &dummy1Ptr, &dummy2Ptr, &tail);  		if (varNsPtr == NULL) { -		    if (flags & TCL_LEAVE_ERR_MSG) { -			VarErrMsg(interp, part1, elName, msg, badNamespace); -		    } -		    goto done; +		    *errMsgPtr = badNamespace; +		    return NULL; +		} else if (tail == NULL) { +		    *errMsgPtr = missingName; +		    return NULL;  		} -		if (tail == NULL) { -		    if (flags & TCL_LEAVE_ERR_MSG) { -			VarErrMsg(interp, part1, elName, msg, missingName); -		    } -		    goto done; +		if (tail != varName) { +		    tailPtr = Tcl_NewStringObj(tail, -1); +		} else { +		    tailPtr = varNamePtr;  		} -		hPtr = Tcl_CreateHashEntry(&varNsPtr->varTable, tail, &new); -		varPtr = NewVar(); -		Tcl_SetHashValue(hPtr, varPtr); -		varPtr->hPtr = hPtr; -		varPtr->nsPtr = varNsPtr; -	    } else {		/* var wasn't found and not to create it */ -		if (flags & TCL_LEAVE_ERR_MSG) { -		    VarErrMsg(interp, part1, elName, msg, noSuchVar); +		varPtr = VarHashCreateVar(&varNsPtr->varTable, tailPtr, +			&isNew); +		if (lookGlobal) { +		    /* +		     * The variable was created starting from the global +		     * namespace: a global reference is returned even if it +		     * wasn't explicitly requested. +		     */ + +		    *indexPtr = -1; +		} else { +		    *indexPtr = -2;  		} -		goto done; +	    } else {		/* Var wasn't found and not to create it. */ +		*errMsgPtr = noSuchVar; +		return NULL;  	    }  	} -    } else {			/* local var: look in frame varFramePtr */ -	Proc *procPtr = varFramePtr->procPtr; -	int localCt = procPtr->numCompiledLocals; -	CompiledLocal *localPtr = procPtr->firstLocalPtr; -	Var *localVarPtr = varFramePtr->compiledLocals; -	int part1Len = strlen(part1); -	 -	for (i = 0;  i < localCt;  i++) { -	    if (!TclIsVarTemporary(localPtr)) { -		register char *localName = localVarPtr->name; -		if ((part1[0] == localName[0]) -		        && (part1Len == localPtr->nameLength) -		        && (strcmp(part1, localName) == 0)) { -		    varPtr = localVarPtr; -		    break; +    } else {			/* Local var: look in frame varFramePtr. */ +	int localLen, localCt = varFramePtr->numCompiledLocals; +	Tcl_Obj **objPtrPtr = &varFramePtr->localCachePtr->varName0; +	const char *localNameStr; + +	for (i=0 ; i<localCt ; i++, objPtrPtr++) { +	    register Tcl_Obj *objPtr = *objPtrPtr; + +	    if (objPtr) { +		localNameStr = TclGetStringFromObj(objPtr, &localLen); + +		if ((varLen == localLen) && (varName[0] == localNameStr[0]) +			&& !memcmp(varName, localNameStr, varLen)) { +		    *indexPtr = i; +		    return (Var *) &varFramePtr->compiledLocals[i];  		}  	    } -	    localVarPtr++; -	    localPtr = localPtr->nextPtr; -	} -	if (varPtr == NULL) {	/* look in the frame's var hash table */ -	    tablePtr = varFramePtr->varTablePtr; -	    if (createPart1) { -		if (tablePtr == NULL) { -		    tablePtr = (Tcl_HashTable *) -			ckalloc(sizeof(Tcl_HashTable)); -		    Tcl_InitHashTable(tablePtr, TCL_STRING_KEYS); -		    varFramePtr->varTablePtr = tablePtr; -		} -		hPtr = Tcl_CreateHashEntry(tablePtr, part1, &new); -		if (new) { -		    varPtr = NewVar(); -		    Tcl_SetHashValue(hPtr, varPtr); -		    varPtr->hPtr = hPtr; -                    varPtr->nsPtr = NULL; /* a local variable */ -		} else { -		    varPtr = (Var *) Tcl_GetHashValue(hPtr); -		} -	    } else { -		hPtr = NULL; -		if (tablePtr != NULL) { -		    hPtr = Tcl_FindHashEntry(tablePtr, part1); -		} -		if (hPtr == NULL) { -		    if (flags & TCL_LEAVE_ERR_MSG) { -			VarErrMsg(interp, part1, elName, msg, noSuchVar); -		    } -		    goto done; -		} -		varPtr = (Var *) Tcl_GetHashValue(hPtr); +	} +	tablePtr = varFramePtr->varTablePtr; +	if (create) { +	    if (tablePtr == NULL) { +		tablePtr = ckalloc(sizeof(TclVarHashTable)); +		TclInitVarHashTable(tablePtr, NULL); +		varFramePtr->varTablePtr = tablePtr; +	    } +	    varPtr = VarHashCreateVar(tablePtr, varNamePtr, &isNew); +	} else { +	    varPtr = NULL; +	    if (tablePtr != NULL) { +		varPtr = VarHashFindVar(tablePtr, varNamePtr); +	    } +	    if (varPtr == NULL) { +		*errMsgPtr = noSuchVar;  	    }  	}      } +    return varPtr; +} + +/* + *---------------------------------------------------------------------- + * + * TclLookupArrayElement -- + * + *	This function is used to locate a variable which is in an array's + *	hashtable given a pointer to the array's Var structure and the + *	element's name. + * + * Results: + *	The return value is a pointer to the variable structure , or NULL if + *	the variable couldn't be found. + * + *	If arrayPtr points to a variable that isn't an array and createPart1 + *	is 1, the corresponding variable will be converted to an array. + *	Otherwise, NULL is returned and an error message is left in the + *	interp's result if TCL_LEAVE_ERR_MSG is set in flags. + * + *	If the variable is not found and createPart2 is 1, the variable is + *	created. Otherwise, NULL is returned and an error message is left in + *	the interp's result if TCL_LEAVE_ERR_MSG is set in flags. + * + *	Note: it's possible for the variable returned to be VAR_UNDEFINED even + *	if createPart1 or createPart2 are 1 (these only cause the hash table + *	entry or array to be created). For example, the variable might be a + *	global that has been unset but is still referenced by a procedure, or + *	a variable that has been unset but it only being kept in existence (if + *	VAR_UNDEFINED) by a trace. + * + * Side effects: + *	The variable at arrayPtr may be converted to be an array if + *	createPart1 is 1. A new hashtable entry may be created if createPart2 + *	is 1. + *	When createElem is 1, callers must incr elNamePtr if they plan + *	to decr it. + * + *---------------------------------------------------------------------- + */ -    lookupVarPart2: -    /* -     * If varPtr is a link variable, we have a reference to some variable -     * that was created through an "upvar" or "global" command. Traverse -     * through any links until we find the referenced variable. -     */ -	 -    while (TclIsVarLink(varPtr)) { -	varPtr = varPtr->value.linkPtr; -    } - -    /* -     * If we're not dealing with an array element, return varPtr. -     */ -     -    if (elName == NULL) { -        goto done; -    } +Var * +TclLookupArrayElement( +    Tcl_Interp *interp,		/* Interpreter to use for lookup. */ +    Tcl_Obj *arrayNamePtr,	/* This is the name of the array, or NULL if +				 * index>= 0. */ +    Tcl_Obj *elNamePtr,		/* Name of element within array. */ +    const int flags,		/* Only TCL_LEAVE_ERR_MSG bit matters. */ +    const char *msg,		/* Verb to use in error messages, e.g. "read" +				 * or "set". Only needed if TCL_LEAVE_ERR_MSG +				 * is set in flags. */ +    const int createArray,	/* If 1, transform arrayName to be an array if +				 * it isn't one yet and the transformation is +				 * possible. If 0, return error if it isn't +				 * already an array. */ +    const int createElem,	/* If 1, create hash table entry for the +				 * element, if it doesn't already exist. If 0, +				 * return error if it doesn't exist. */ +    Var *arrayPtr,		/* Pointer to the array's Var structure. */ +    int index)			/* If >=0, the index of the local array. */ +{ +    int isNew; +    Var *varPtr; +    TclVarHashTable *tablePtr; +    Namespace *nsPtr;      /* -     * We're dealing with an array element. Make sure the variable is an -     * array and look up the element (create the element if desired). +     * We're dealing with an array element. Make sure the variable is an array +     * and look up the element (create the element if desired).       */ -    if (TclIsVarUndefined(varPtr) && !TclIsVarArrayElement(varPtr)) { -	if (!createPart1) { +    if (TclIsVarUndefined(arrayPtr) && !TclIsVarArrayElement(arrayPtr)) { +	if (!createArray) {  	    if (flags & TCL_LEAVE_ERR_MSG) { -		VarErrMsg(interp, part1, elName, msg, noSuchVar); +		TclObjVarErrMsg(interp, arrayNamePtr, elNamePtr, msg, +			noSuchVar, index); +		Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", +			arrayNamePtr?TclGetString(arrayNamePtr):NULL, NULL);  	    } -	    varPtr = NULL; -	    goto done; +	    return NULL;  	}  	/*  	 * Make sure we are not resurrecting a namespace variable from a  	 * deleted namespace!  	 */ -	if ((varPtr->flags & VAR_IN_HASHTABLE) && (varPtr->hPtr == NULL)) { + +	if (TclIsVarDeadHash(arrayPtr)) {  	    if (flags & TCL_LEAVE_ERR_MSG) { -		VarErrMsg(interp, part1, elName, msg, danglingVar); +		TclObjVarErrMsg(interp, arrayNamePtr, elNamePtr, msg, +			danglingVar, index); +		Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", +			arrayNamePtr?TclGetString(arrayNamePtr):NULL, NULL);  	    } -	    varPtr = NULL; -	    goto done; +	    return NULL;  	} -	TclSetVarArray(varPtr); -	TclClearVarUndefined(varPtr); -	varPtr->value.tablePtr = -	    (Tcl_HashTable *) ckalloc(sizeof(Tcl_HashTable)); -	Tcl_InitHashTable(varPtr->value.tablePtr, TCL_STRING_KEYS); -    } else if (!TclIsVarArray(varPtr)) { +	TclSetVarArray(arrayPtr); +	tablePtr = ckalloc(sizeof(TclVarHashTable)); +	arrayPtr->value.tablePtr = tablePtr; + +	if (TclIsVarInHash(arrayPtr) && TclGetVarNsPtr(arrayPtr)) { +	    nsPtr = TclGetVarNsPtr(arrayPtr); +	} else { +	    nsPtr = NULL; +	} +	TclInitVarHashTable(arrayPtr->value.tablePtr, nsPtr); +    } else if (!TclIsVarArray(arrayPtr)) {  	if (flags & TCL_LEAVE_ERR_MSG) { -	    VarErrMsg(interp, part1, elName, msg, needArray); +	    TclObjVarErrMsg(interp, arrayNamePtr, elNamePtr, msg, needArray, +		    index); +	    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", +		    arrayNamePtr?TclGetString(arrayNamePtr):NULL, NULL);  	} -	varPtr = NULL; -	goto done; +	return NULL;      } -    *arrayPtrPtr = varPtr; -    if (createPart2) { -	hPtr = Tcl_CreateHashEntry(varPtr->value.tablePtr, elName, &new); -	if (new) { -	    if (varPtr->searchPtr != NULL) { -		DeleteSearches(varPtr); + +    if (createElem) { +	varPtr = VarHashCreateVar(arrayPtr->value.tablePtr, elNamePtr, +		&isNew); +	if (isNew) { +	    if (arrayPtr->flags & VAR_SEARCH_ACTIVE) { +		DeleteSearches((Interp *) interp, arrayPtr);  	    } -	    varPtr = NewVar(); -	    Tcl_SetHashValue(hPtr, varPtr); -	    varPtr->hPtr = hPtr; -	    varPtr->nsPtr = varNsPtr;  	    TclSetVarArrayElement(varPtr);  	}      } else { -	hPtr = Tcl_FindHashEntry(varPtr->value.tablePtr, elName); -	if (hPtr == NULL) { +	varPtr = VarHashFindVar(arrayPtr->value.tablePtr, elNamePtr); +	if (varPtr == NULL) {  	    if (flags & TCL_LEAVE_ERR_MSG) { -		VarErrMsg(interp, part1, elName, msg, noSuchElement); +		TclObjVarErrMsg(interp, arrayNamePtr, elNamePtr, msg, +			noSuchElement, index); +		Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ELEMENT", +			TclGetString(elNamePtr), NULL);  	    } -	    varPtr = NULL; -	    goto done;  	}      } -    varPtr = (Var *) Tcl_GetHashValue(hPtr); - -    done: -    if (openParen != NULL) { -        *openParen = '('; -	*closeParen = ')'; -    }      return varPtr;  } @@ -461,9 +1234,9 @@ TclLookupVar(interp, part1, part2, flags, msg, createPart1, createPart2,   *   * Results:   *	The return value points to the current value of varName as a string. - *	If the variable is not defined or can't be read because of a clash - *	in array usage then a NULL pointer is returned and an error message - *	is left in the interp's result if the TCL_LEAVE_ERR_MSG flag is set. + *	If the variable is not defined or can't be read because of a clash in + *	array usage then a NULL pointer is returned and an error message is + *	left in the interp's result if the TCL_LEAVE_ERR_MSG flag is set.   *	Note: the return value is only valid up until the next change to the   *	variable; if you depend on the value lasting longer than that, then   *	make yourself a private copy. @@ -474,16 +1247,25 @@ TclLookupVar(interp, part1, part2, flags, msg, createPart1, createPart2,   *----------------------------------------------------------------------   */ -CONST char * -Tcl_GetVar(interp, varName, flags) -    Tcl_Interp *interp;		/* Command interpreter in which varName is -				 * to be looked up. */ -    char *varName;		/* Name of a variable in interp. */ -    int flags;			/* OR-ed combination of TCL_GLOBAL_ONLY, +#undef Tcl_GetVar +const char * +Tcl_GetVar( +    Tcl_Interp *interp,		/* Command interpreter in which varName is to +				 * be looked up. */ +    const char *varName,	/* Name of a variable in interp. */ +    int flags)			/* OR-ed combination of TCL_GLOBAL_ONLY,  				 * TCL_NAMESPACE_ONLY or TCL_LEAVE_ERR_MSG  				 * bits. */  { -    return Tcl_GetVar2(interp, varName, (char *) NULL, flags); +    Tcl_Obj *varNamePtr = Tcl_NewStringObj(varName, -1); +    Tcl_Obj *resultPtr = Tcl_ObjGetVar2(interp, varNamePtr, NULL, flags); + +    TclDecrRefCount(varNamePtr); + +    if (resultPtr == NULL) { +	return NULL; +    } +    return TclGetString(resultPtr);  }  /* @@ -491,17 +1273,17 @@ Tcl_GetVar(interp, varName, flags)   *   * Tcl_GetVar2 --   * - *	Return the value of a Tcl variable as a string, given a two-part - *	name consisting of array name and element within array. + *	Return the value of a Tcl variable as a string, given a two-part name + *	consisting of array name and element within array.   *   * Results: - *	The return value points to the current value of the variable given - *	by part1 and part2 as a string. If the specified variable doesn't - *	exist, or if there is a clash in array usage, then NULL is returned - *	and a message will be left in the interp's result if the - *	TCL_LEAVE_ERR_MSG flag is set. Note: the return value is only valid - *	up until the next change to the variable; if you depend on the value - *	lasting longer than that, then make yourself a private copy. + *	The return value points to the current value of the variable given by + *	part1 and part2 as a string. If the specified variable doesn't exist, + *	or if there is a clash in array usage, then NULL is returned and a + *	message will be left in the interp's result if the TCL_LEAVE_ERR_MSG + *	flag is set. Note: the return value is only valid up until the next + *	change to the variable; if you depend on the value lasting longer than + *	that, then make yourself a private copy.   *   * Side effects:   *	None. @@ -509,72 +1291,36 @@ Tcl_GetVar(interp, varName, flags)   *----------------------------------------------------------------------   */ -CONST char * -Tcl_GetVar2(interp, part1, part2, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be looked up. */ -    char *part1;		/* Name of an array (if part2 is non-NULL) -				 * or the name of a variable. */ -    char *part2;		/* If non-NULL, gives the name of an element +const char * +Tcl_GetVar2( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be looked up. */ +    const char *part1,		/* Name of an array (if part2 is non-NULL) or +				 * the name of a variable. */ +    const char *part2,		/* If non-NULL, gives the name of an element  				 * in the array part1. */ -    int flags;			/* OR-ed combination of TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY and TCL_LEAVE_ERR_MSG -                                 * bits. */ +    int flags)			/* OR-ed combination of TCL_GLOBAL_ONLY, +				 * TCL_NAMESPACE_ONLY and TCL_LEAVE_ERR_MSG * +				 * bits. */  { -    Tcl_Obj *objPtr; +    Tcl_Obj *resultPtr; +    Tcl_Obj *part2Ptr = NULL, *part1Ptr = Tcl_NewStringObj(part1, -1); -    objPtr = Tcl_GetVar2Ex(interp, part1, part2, flags); -    if (objPtr == NULL) { -	return NULL; +    if (part2) { +	part2Ptr = Tcl_NewStringObj(part2, -1); +	Tcl_IncrRefCount(part2Ptr);      } -    return TclGetString(objPtr); -} -/* - *---------------------------------------------------------------------- - * - * Tcl_ObjGetVar2 -- - * - *	Return the value of a Tcl variable as a Tcl object, given a - *	two-part name consisting of array name and element within array. - * - * Results: - *	The return value points to the current object value of the variable - *	given by part1Ptr and part2Ptr. If the specified variable doesn't - *	exist, or if there is a clash in array usage, then NULL is returned - *	and a message will be left in the interpreter's result if the - *	TCL_LEAVE_ERR_MSG flag is set. - * - * Side effects: - *	The ref count for the returned object is _not_ incremented to - *	reflect the returned reference; if you want to keep a reference to - *	the object you must increment its ref count yourself. - * - *---------------------------------------------------------------------- - */ -Tcl_Obj * -Tcl_ObjGetVar2(interp, part1Ptr, part2Ptr, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be looked up. */ -    register Tcl_Obj *part1Ptr;	/* Points to an object holding the name of -				 * an array (if part2 is non-NULL) or the -				 * name of a variable. */ -    register Tcl_Obj *part2Ptr;	/* If non-null, points to an object holding -				 * the name of an element in the array -				 * part1Ptr. */ -    int flags;			/* OR-ed combination of TCL_GLOBAL_ONLY and -				 * TCL_LEAVE_ERR_MSG bits. */ -{ -    char *part1, *part2; +    resultPtr = Tcl_ObjGetVar2(interp, part1Ptr, part2Ptr, flags); -    part1 = Tcl_GetString(part1Ptr); -    if (part2Ptr != NULL) { -	part2 = Tcl_GetString(part2Ptr); -    } else { -	part2 = NULL; +    Tcl_DecrRefCount(part1Ptr); +    if (part2Ptr) { +	Tcl_DecrRefCount(part2Ptr);      } -     -    return Tcl_GetVar2Ex(interp, part1, part2, flags); +    if (resultPtr == NULL) { +	return NULL; +    } +    return TclGetString(resultPtr);  }  /* @@ -582,8 +1328,8 @@ Tcl_ObjGetVar2(interp, part1Ptr, part2Ptr, flags)   *   * Tcl_GetVar2Ex --   * - *	Return the value of a Tcl variable as a Tcl object, given a - *	two-part name consisting of array name and element within array. + *	Return the value of a Tcl variable as a Tcl object, given a two-part + *	name consisting of array name and element within array.   *   * Results:   *	The return value points to the current object value of the variable @@ -593,370 +1339,148 @@ Tcl_ObjGetVar2(interp, part1Ptr, part2Ptr, flags)   *	TCL_LEAVE_ERR_MSG flag is set.   *   * Side effects: - *	The ref count for the returned object is _not_ incremented to - *	reflect the returned reference; if you want to keep a reference to - *	the object you must increment its ref count yourself. + *	The ref count for the returned object is _not_ incremented to reflect + *	the returned reference; if you want to keep a reference to the object + *	you must increment its ref count yourself.   *   *----------------------------------------------------------------------   */  Tcl_Obj * -Tcl_GetVar2Ex(interp, part1, part2, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be looked up. */ -    char *part1;		/* Name of an array (if part2 is non-NULL) -				 * or the name of a variable. */ -    char *part2;		/* If non-NULL, gives the name of an element +Tcl_GetVar2Ex( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be looked up. */ +    const char *part1,		/* Name of an array (if part2 is non-NULL) or +				 * the name of a variable. */ +    const char *part2,		/* If non-NULL, gives the name of an element  				 * in the array part1. */ -    int flags;			/* OR-ed combination of TCL_GLOBAL_ONLY, -				 * and TCL_LEAVE_ERR_MSG bits. */ +    int flags)			/* OR-ed combination of TCL_GLOBAL_ONLY, and +				 * TCL_LEAVE_ERR_MSG bits. */  { -    Interp *iPtr = (Interp *) interp; -    register Var *varPtr; -    Var *arrayPtr; -    char *msg; +    Tcl_Obj *resPtr, *part2Ptr = NULL, *part1Ptr = Tcl_NewStringObj(part1, -1); -    /* -     * We need a special flag check to see if we want to create part 1, -     * because commands like lappend require read traces to trigger for -     * previously non-existent values. -     */ -    varPtr = TclLookupVar(interp, part1, part2, flags, "read", -            /*createPart1*/ (flags & TCL_TRACE_READS), -	    /*createPart2*/ 1, &arrayPtr); -    if (varPtr == NULL) { -	return NULL; +    if (part2) { +	part2Ptr = Tcl_NewStringObj(part2, -1); +	Tcl_IncrRefCount(part2Ptr);      } -    /* -     * Invoke any traces that have been set for the variable. -     */ - -    if ((varPtr->tracePtr != NULL) -	    || ((arrayPtr != NULL) && (arrayPtr->tracePtr != NULL))) { -	int resultType; -	msg = CallTraces(iPtr, arrayPtr, varPtr, part1, part2, -		(flags & (TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY)) | TCL_TRACE_READS, -		&resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, part1, part2, "read", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, part1, part2, "read", msg); -		} -	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *) msg); -	    } -	    goto errorReturn; -	} -    } +    resPtr = Tcl_ObjGetVar2(interp, part1Ptr, part2Ptr, flags); -    /* -     * Return the element if it's an existing scalar variable. -     */ -     -    if (TclIsVarScalar(varPtr) && !TclIsVarUndefined(varPtr)) { -	return varPtr->value.objPtr; -    } -     -    if (flags & TCL_LEAVE_ERR_MSG) { -	if (TclIsVarUndefined(varPtr) && (arrayPtr != NULL) -	        && !TclIsVarUndefined(arrayPtr)) { -	    msg = noSuchElement; -	} else if (TclIsVarArray(varPtr)) { -	    msg = isArray; -	} else { -	    msg = noSuchVar; -	} -	VarErrMsg(interp, part1, part2, "read", msg); +    Tcl_DecrRefCount(part1Ptr); +    if (part2Ptr) { +	Tcl_DecrRefCount(part2Ptr);      } -    /* -     * An error. If the variable doesn't exist anymore and no-one's using -     * it, then free up the relevant structures and hash table entries. -     */ - -    errorReturn: -    if (TclIsVarUndefined(varPtr)) { -	CleanupVar(varPtr, arrayPtr); -    } -    return NULL; +    return resPtr;  }  /*   *----------------------------------------------------------------------   * - * TclGetIndexedScalar -- + * Tcl_ObjGetVar2 --   * - *	Return the Tcl object value of a local scalar variable in the active - *	procedure, given its index in the procedure's array of compiler - *	allocated local variables. + *	Return the value of a Tcl variable as a Tcl object, given a two-part + *	name consisting of array name and element within array.   *   * Results:   *	The return value points to the current object value of the variable - *	given by localIndex. If the specified variable doesn't exist, or - *	there is a clash in array usage, or an error occurs while executing - *	variable traces, then NULL is returned and a message will be left in - *	the interpreter's result if TCL_LEAVE_ERR_MSG is set in flags. + *	given by part1Ptr and part2Ptr. If the specified variable doesn't + *	exist, or if there is a clash in array usage, then NULL is returned + *	and a message will be left in the interpreter's result if the + *	TCL_LEAVE_ERR_MSG flag is set.   *   * Side effects: - *	The ref count for the returned object is _not_ incremented to - *	reflect the returned reference; if you want to keep a reference to - *	the object you must increment its ref count yourself. + *	The ref count for the returned object is _not_ incremented to reflect + *	the returned reference; if you want to keep a reference to the object + *	you must increment its ref count yourself. + * + *	Callers must incr part2Ptr if they plan to decr it.   *   *----------------------------------------------------------------------   */  Tcl_Obj * -TclGetIndexedScalar(interp, localIndex, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be looked up. */ -    register int localIndex;	/* Index of variable in procedure's array -				 * of local variables. */ -    int flags;			/* TCL_LEAVE_ERR_MSG if to leave an error -				 * message in interpreter's result on an error. -				 * Otherwise no error message is left. */ +Tcl_ObjGetVar2( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be looked up. */ +    register Tcl_Obj *part1Ptr,	/* Points to an object holding the name of an +				 * array (if part2 is non-NULL) or the name of +				 * a variable. */ +    register Tcl_Obj *part2Ptr,	/* If non-null, points to an object holding +				 * the name of an element in the array +				 * part1Ptr. */ +    int flags)			/* OR-ed combination of TCL_GLOBAL_ONLY and +				 * TCL_LEAVE_ERR_MSG bits. */  { -    Interp *iPtr = (Interp *) interp; -    CallFrame *varFramePtr = iPtr->varFramePtr; -				/* Points to the procedure call frame whose -				 * variables are currently in use. Same as -				 * the current procedure's frame, if any, -				 * unless an "uplevel" is executing. */ -    Var *compiledLocals = varFramePtr->compiledLocals; -    register Var *varPtr;	/* Points to the variable's in-frame Var -				 * structure. */ -    char *varName;		/* Name of the local variable. */ -    char *msg; - -#ifdef TCL_COMPILE_DEBUG -    int localCt = varFramePtr->procPtr->numCompiledLocals; - -    if (compiledLocals == NULL) { -	fprintf(stderr, "\nTclGetIndexedScalar: can't get local %i in frame 0x%x, no compiled locals\n", -		localIndex, (unsigned int) varFramePtr); -	panic("TclGetIndexedScalar: no compiled locals in frame 0x%x", -		(unsigned int) varFramePtr); -    } -    if ((localIndex < 0) || (localIndex >= localCt)) { -	fprintf(stderr, "\nTclGetIndexedScalar: can't get local %i in frame 0x%x with %i locals\n", -		localIndex, (unsigned int) varFramePtr, localCt); -	panic("TclGetIndexedScalar: bad local index %i in frame 0x%x", -		localIndex, (unsigned int) varFramePtr); -    } -#endif /* TCL_COMPILE_DEBUG */ -     -    varPtr = &(compiledLocals[localIndex]); -    varName = varPtr->name; - -    /* -     * If varPtr is a link variable, we have a reference to some variable -     * that was created through an "upvar" or "global" command, or we have a -     * reference to a variable in an enclosing namespace. Traverse through -     * any links until we find the referenced variable. -     */ -	 -    while (TclIsVarLink(varPtr)) { -	varPtr = varPtr->value.linkPtr; -    } - -    /* -     * Invoke any traces that have been set for the variable. -     */ - -    if (varPtr->tracePtr != NULL) { -	int resultType; - -	msg = CallTraces(iPtr, /*arrayPtr*/ NULL, varPtr, varName, NULL, -		TCL_TRACE_READS, &resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, varName, NULL, "read", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, varName, NULL, "read", msg); -		} -	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *) msg); -	    } -	    return NULL; -	} -    } +    Var *varPtr, *arrayPtr;      /* -     * Make sure we're dealing with a scalar variable and not an array, and -     * that the variable exists (isn't undefined). +     * Filter to pass through only the flags this interface supports.       */ -    if (!TclIsVarScalar(varPtr) || TclIsVarUndefined(varPtr)) { -	if (flags & TCL_LEAVE_ERR_MSG) { -	    if (TclIsVarArray(varPtr)) { -		msg = isArray; -	    } else { -		msg = noSuchVar; -	    } -	    VarErrMsg(interp, varName, NULL, "read", msg); -	} +    flags &= (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY|TCL_LEAVE_ERR_MSG); +    varPtr = TclObjLookupVarEx(interp, part1Ptr, part2Ptr, flags, "read", +	    /*createPart1*/ 0, /*createPart2*/ 1, &arrayPtr); +    if (varPtr == NULL) {  	return NULL;      } -    return varPtr->value.objPtr; + +    return TclPtrGetVar(interp, varPtr, arrayPtr, part1Ptr, part2Ptr, +	    flags, -1);  }  /*   *----------------------------------------------------------------------   * - * TclGetElementOfIndexedArray -- + * TclPtrGetVar --   * - *	Return the Tcl object value for an element in a local array - *	variable. The element is named by the object elemPtr while the  - *	array is specified by its index in the active procedure's array - *	of compiler allocated local variables. + *	Return the value of a Tcl variable as a Tcl object, given the pointers + *	to the variable's (and possibly containing array's) VAR structure.   *   * Results: - *	The return value points to the current object value of the - *	element. If the specified array or element doesn't exist, or there - *	is a clash in array usage, or an error occurs while executing - *	variable traces, then NULL is returned and a message will be left in - *	the interpreter's result if TCL_LEAVE_ERR_MSG is set in flags. + *	The return value points to the current object value of the variable + *	given by varPtr. If the specified variable doesn't exist, or if there + *	is a clash in array usage, then NULL is returned and a message will be + *	left in the interpreter's result if the TCL_LEAVE_ERR_MSG flag is set.   *   * Side effects: - *	The ref count for the returned object is _not_ incremented to - *	reflect the returned reference; if you want to keep a reference to - *	the object you must increment its ref count yourself. + *	The ref count for the returned object is _not_ incremented to reflect + *	the returned reference; if you want to keep a reference to the object + *	you must increment its ref count yourself.   *   *----------------------------------------------------------------------   */  Tcl_Obj * -TclGetElementOfIndexedArray(interp, localIndex, elemPtr, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be looked up. */ -    int localIndex;		/* Index of array variable in procedure's -				 * array of local variables. */ -    Tcl_Obj *elemPtr;		/* Points to an object holding the name of -				 * an element to get in the array. */ -    int flags;			/* TCL_LEAVE_ERR_MSG if to leave an error -				 * message in interpreter's result on an error. -				 * Otherwise no error message is left. */ +TclPtrGetVar( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be looked up. */ +    register Var *varPtr,	/* The variable to be read.*/ +    Var *arrayPtr,		/* NULL for scalar variables, pointer to the +				 * containing array otherwise. */ +    Tcl_Obj *part1Ptr,		/* Name of an array (if part2 is non-NULL) or +				 * the name of a variable. */ +    Tcl_Obj *part2Ptr,		/* If non-NULL, gives the name of an element +				 * in the array part1. */ +    const int flags,		/* OR-ed combination of TCL_GLOBAL_ONLY, and +				 * TCL_LEAVE_ERR_MSG bits. */ +    int index)			/* Index into the local variable table of the +				 * variable, or -1. Only used when part1Ptr is +				 * NULL. */  {      Interp *iPtr = (Interp *) interp; -    CallFrame *varFramePtr = iPtr->varFramePtr; -				/* Points to the procedure call frame whose -				 * variables are currently in use. Same as -				 * the current procedure's frame, if any, -				 * unless an "uplevel" is executing. */ -    Var *compiledLocals = varFramePtr->compiledLocals; -    Var *arrayPtr;		/* Points to the array's in-frame Var -				 * structure. */ -    char *arrayName;		/* Name of the local array. */ -    Tcl_HashEntry *hPtr; -    Var *varPtr = NULL;		/* Points to the element's Var structure -				 * that we return. Initialized to avoid -				 * compiler warning. */ -    char *elem, *msg; -    int new; - -#ifdef TCL_COMPILE_DEBUG -    Proc *procPtr = varFramePtr->procPtr; -    int localCt = procPtr->numCompiledLocals; - -    if (compiledLocals == NULL) { -	fprintf(stderr, "\nTclGetElementOfIndexedArray: can't get element of local %i in frame 0x%x, no compiled locals\n", -		localIndex, (unsigned int) varFramePtr); -	panic("TclGetIndexedScalar: no compiled locals in frame 0x%x", -		(unsigned int) varFramePtr); -    } -    if ((localIndex < 0) || (localIndex >= localCt)) { -	fprintf(stderr, "\nTclGetIndexedScalar: can't get element of local %i in frame 0x%x with %i locals\n", -		localIndex, (unsigned int) varFramePtr, localCt); -	panic("TclGetElementOfIndexedArray: bad local index %i in frame 0x%x", -		localIndex, (unsigned int) varFramePtr); -    } -#endif /* TCL_COMPILE_DEBUG */ - -    elem = TclGetString(elemPtr); -    arrayPtr = &(compiledLocals[localIndex]); -    arrayName = arrayPtr->name; - -    /* -     * If arrayPtr is a link variable, we have a reference to some variable -     * that was created through an "upvar" or "global" command, or we have a -     * reference to a variable in an enclosing namespace. Traverse through -     * any links until we find the referenced variable. -     */ -	 -    while (TclIsVarLink(arrayPtr)) { -	arrayPtr = arrayPtr->value.linkPtr; -    } - -    /* -     * Make sure we're dealing with an array and that the array variable -     * exists (isn't undefined). -     */ - -    if (!TclIsVarArray(arrayPtr) || TclIsVarUndefined(arrayPtr)) { -	if (flags & TCL_LEAVE_ERR_MSG) { -	    VarErrMsg(interp, arrayName, elem, "read", noSuchVar); -	} -	goto errorReturn; -    }  - -    /* -     * Look up the element. Note that we must create the element (but leave -     * it marked undefined) if it does not already exist. This allows a -     * trace to create new array elements "on the fly" that did not exist -     * before. A trace is always passed a variable for the array element. If -     * the trace does not define the variable, it will be deleted below (at -     * errorReturn) and an error returned. -     */ - -    hPtr = Tcl_CreateHashEntry(arrayPtr->value.tablePtr, elem, &new); -    if (new) { -	if (arrayPtr->searchPtr != NULL) { -	    DeleteSearches(arrayPtr); -	} -	varPtr = NewVar(); -	Tcl_SetHashValue(hPtr, varPtr); -	varPtr->hPtr = hPtr; -	varPtr->nsPtr = varFramePtr->nsPtr; -	TclSetVarArrayElement(varPtr); -    } else { -	varPtr = (Var *) Tcl_GetHashValue(hPtr); -    } +    const char *msg;      /* -     * Invoke any traces that have been set for the element variable. +     * Invoke any read traces that have been set for the variable.       */ -    if ((varPtr->tracePtr != NULL) -            || ((arrayPtr != NULL) && (arrayPtr->tracePtr != NULL))) { -	int resultType; - -	msg = CallTraces(iPtr, arrayPtr, varPtr, arrayName, elem, -	        TCL_TRACE_READS, &resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, arrayName, elem, "read", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, arrayName, elem, "read", msg); -		} -	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *)msg); -	    } +    if ((varPtr->flags & VAR_TRACED_READ) +	    || (arrayPtr && (arrayPtr->flags & VAR_TRACED_READ))) { +	if (TCL_ERROR == TclObjCallVarTraces(iPtr, arrayPtr, varPtr, +		part1Ptr, part2Ptr, +		(flags & (TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY)) +		| TCL_TRACE_READS, (flags & TCL_LEAVE_ERR_MSG), index)) {  	    goto errorReturn;  	}      } @@ -964,28 +1488,32 @@ TclGetElementOfIndexedArray(interp, localIndex, elemPtr, flags)      /*       * Return the element if it's an existing scalar variable.       */ -     +      if (TclIsVarScalar(varPtr) && !TclIsVarUndefined(varPtr)) {  	return varPtr->value.objPtr;      } -     +      if (flags & TCL_LEAVE_ERR_MSG) { -	if (TclIsVarArray(varPtr)) { +	if (TclIsVarUndefined(varPtr) && arrayPtr +		&& !TclIsVarUndefined(arrayPtr)) { +	    msg = noSuchElement; +	} else if (TclIsVarArray(varPtr)) {  	    msg = isArray;  	} else {  	    msg = noSuchVar;  	} -	VarErrMsg(interp, arrayName, elem, "read", msg); +	TclObjVarErrMsg(interp, part1Ptr, part2Ptr, "read", msg, index);      }      /* -     * An error. If the variable doesn't exist anymore and no-one's using -     * it, then free up the relevant structures and hash table entries. +     * An error. If the variable doesn't exist anymore and no-one's using it, +     * then free up the relevant structures and hash table entries.       */ -    errorReturn: -    if ((varPtr != NULL) && TclIsVarUndefined(varPtr)) { -	CleanupVar(varPtr, NULL); /* the array is not in a hashtable */ +  errorReturn: +    Tcl_SetErrorCode(interp, "TCL", "READ", "VARNAME", NULL); +    if (TclIsVarUndefined(varPtr)) { +	TclCleanupVar(varPtr, arrayPtr);      }      return NULL;  } @@ -995,8 +1523,8 @@ TclGetElementOfIndexedArray(interp, localIndex, elemPtr, flags)   *   * Tcl_SetObjCmd --   * - *	This procedure is invoked to process the "set" Tcl command. - *	See the user documentation for details on what it does. + *	This function is invoked to process the "set" Tcl command. See the + *	user documentation for details on what it does.   *   * Results:   *	A standard Tcl result value. @@ -1009,23 +1537,22 @@ TclGetElementOfIndexedArray(interp, localIndex, elemPtr, flags)  	/* ARGSUSED */  int -Tcl_SetObjCmd(dummy, interp, objc, objv) -    ClientData dummy;			/* Not used. */ -    register Tcl_Interp *interp;	/* Current interpreter. */ -    int objc;				/* Number of arguments. */ -    Tcl_Obj *CONST objv[];		/* Argument objects. */ +Tcl_SetObjCmd( +    ClientData dummy,		/* Not used. */ +    register Tcl_Interp *interp,/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  {      Tcl_Obj *varValueObj;      if (objc == 2) { -	varValueObj = Tcl_ObjGetVar2(interp, objv[1], NULL, TCL_LEAVE_ERR_MSG); +	varValueObj = Tcl_ObjGetVar2(interp, objv[1], NULL,TCL_LEAVE_ERR_MSG);  	if (varValueObj == NULL) {  	    return TCL_ERROR;  	}  	Tcl_SetObjResult(interp, varValueObj);  	return TCL_OK;      } else if (objc == 3) { -  	varValueObj = Tcl_ObjSetVar2(interp, objv[1], NULL, objv[2],  		TCL_LEAVE_ERR_MSG);  	if (varValueObj == NULL) { @@ -1048,33 +1575,44 @@ Tcl_SetObjCmd(dummy, interp, objc, objv)   *   * Results:   *	Returns a pointer to the malloc'ed string which is the character - *	representation of the variable's new value. The caller must not - *	modify this string. If the write operation was disallowed then NULL - *	is returned; if the TCL_LEAVE_ERR_MSG flag is set, then an - *	explanatory message will be left in the interp's result. Note that the - *	returned string may not be the same as newValue; this is because - *	variable traces may modify the variable's value. + *	representation of the variable's new value. The caller must not modify + *	this string. If the write operation was disallowed then NULL is + *	returned; if the TCL_LEAVE_ERR_MSG flag is set, then an explanatory + *	message will be left in the interp's result. Note that the returned + *	string may not be the same as newValue; this is because variable + *	traces may modify the variable's value.   *   * Side effects: - *	If varName is defined as a local or global variable in interp, - *	its value is changed to newValue. If varName isn't currently - *	defined, then a new global variable by that name is created. + *	If varName is defined as a local or global variable in interp, its + *	value is changed to newValue. If varName isn't currently defined, then + *	a new global variable by that name is created.   *   *----------------------------------------------------------------------   */ -CONST char * -Tcl_SetVar(interp, varName, newValue, flags) -    Tcl_Interp *interp;		/* Command interpreter in which varName is -				 * to be looked up. */ -    char *varName;		/* Name of a variable in interp. */ -    CONST char *newValue;	/* New value for varName. */ -    int flags;			/* Various flags that tell how to set value: -				 * any of TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY, TCL_APPEND_VALUE, -				 * TCL_LIST_ELEMENT, TCL_LEAVE_ERR_MSG. */ +#undef Tcl_SetVar +const char * +Tcl_SetVar( +    Tcl_Interp *interp,		/* Command interpreter in which varName is to +				 * be looked up. */ +    const char *varName,	/* Name of a variable in interp. */ +    const char *newValue,	/* New value for varName. */ +    int flags)			/* Various flags that tell how to set value: +				 * any of TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_APPEND_VALUE, TCL_LIST_ELEMENT, +				 * TCL_LEAVE_ERR_MSG. */  { -    return Tcl_SetVar2(interp, varName, (char *) NULL, newValue, flags); +    Tcl_Obj *varValuePtr, *varNamePtr = Tcl_NewStringObj(varName, -1); + +    Tcl_IncrRefCount(varNamePtr); +    varValuePtr = Tcl_ObjSetVar2(interp, varNamePtr, NULL,  +	    Tcl_NewStringObj(newValue, -1), flags); +    Tcl_DecrRefCount(varNamePtr); + +    if (varValuePtr == NULL) { +	return NULL; +    } +    return TclGetString(varValuePtr);  }  /* @@ -1082,57 +1620,45 @@ Tcl_SetVar(interp, varName, newValue, flags)   *   * Tcl_SetVar2 --   * - *      Given a two-part variable name, which may refer either to a - *      scalar variable or an element of an array, change the value - *      of the variable.  If the named scalar or array or element - *      doesn't exist then create one. + *	Given a two-part variable name, which may refer either to a scalar + *	variable or an element of an array, change the value of the variable. + *	If the named scalar or array or element doesn't exist then create one.   *   * Results:   *	Returns a pointer to the malloc'ed string which is the character - *	representation of the variable's new value. The caller must not - *	modify this string. If the write operation was disallowed because an - *	array was expected but not found (or vice versa), then NULL is - *	returned; if the TCL_LEAVE_ERR_MSG flag is set, then an explanatory - *	message will be left in the interp's result. Note that the returned - *	string may not be the same as newValue; this is because variable - *	traces may modify the variable's value. + *	representation of the variable's new value. The caller must not modify + *	this string. If the write operation was disallowed because an array + *	was expected but not found (or vice versa), then NULL is returned; if + *	the TCL_LEAVE_ERR_MSG flag is set, then an explanatory message will be + *	left in the interp's result. Note that the returned string may not be + *	the same as newValue; this is because variable traces may modify the + *	variable's value.   *   * Side effects: - *      The value of the given variable is set. If either the array - *      or the entry didn't exist then a new one is created. + *	The value of the given variable is set. If either the array or the + *	entry didn't exist then a new one is created.   *   *----------------------------------------------------------------------   */ -CONST char * -Tcl_SetVar2(interp, part1, part2, newValue, flags) -    Tcl_Interp *interp;         /* Command interpreter in which variable is -                                 * to be looked up. */ -    char *part1;                /* If part2 is NULL, this is name of scalar -                                 * variable. Otherwise it is the name of -                                 * an array. */ -    char *part2;                /* Name of an element within an array, or +const char * +Tcl_SetVar2( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be looked up. */ +    const char *part1,		/* If part2 is NULL, this is name of scalar +				 * variable. Otherwise it is the name of an +				 * array. */ +    const char *part2,		/* Name of an element within an array, or  				 * NULL. */ -    CONST char *newValue;       /* New value for variable. */ -    int flags;                  /* Various flags that tell how to set value: -				 * any of TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY, TCL_APPEND_VALUE, -				 * TCL_LIST_ELEMENT, or TCL_LEAVE_ERR_MSG */ +    const char *newValue,	/* New value for variable. */ +    int flags)			/* Various flags that tell how to set value: +				 * any of TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_APPEND_VALUE, TCL_LIST_ELEMENT, or +				 * TCL_LEAVE_ERR_MSG. */  { -    register Tcl_Obj *valuePtr; -    Tcl_Obj *varValuePtr; +    Tcl_Obj *varValuePtr = Tcl_SetVar2Ex(interp, part1, part2, +	    Tcl_NewStringObj(newValue, -1), flags); -    /* -     * Create an object holding the variable's new value and use -     * Tcl_SetVar2Ex to actually set the variable. -     */ - -    valuePtr = Tcl_NewStringObj(newValue, -1); -    Tcl_IncrRefCount(valuePtr); - -    varValuePtr = Tcl_SetVar2Ex(interp, part1, part2, valuePtr, flags); -    Tcl_DecrRefCount(valuePtr); /* done with the object */ -          if (varValuePtr == NULL) {  	return NULL;      } @@ -1142,190 +1668,278 @@ Tcl_SetVar2(interp, part1, part2, newValue, flags)  /*   *----------------------------------------------------------------------   * - * Tcl_ObjSetVar2 -- + * Tcl_SetVar2Ex --   * - *	This function is the same as Tcl_SetVar2Ex below, except the - *	variable names are passed in Tcl object instead of strings. + *	Given a two-part variable name, which may refer either to a scalar + *	variable or an element of an array, change the value of the variable + *	to a new Tcl object value. If the named scalar or array or element + *	doesn't exist then create one.   *   * Results:   *	Returns a pointer to the Tcl_Obj holding the new value of the   *	variable. If the write operation was disallowed because an array was - *	expected but not found (or vice versa), then NULL is returned; if - *	the TCL_LEAVE_ERR_MSG flag is set, then an explanatory message will - *	be left in the interpreter's result. Note that the returned object - *	may not be the same one referenced by newValuePtr; this is because + *	expected but not found (or vice versa), then NULL is returned; if the + *	TCL_LEAVE_ERR_MSG flag is set, then an explanatory message will be + *	left in the interpreter's result. Note that the returned object may + *	not be the same one referenced by newValuePtr; this is because   *	variable traces may modify the variable's value.   *   * Side effects:   *	The value of the given variable is set. If either the array or the   *	entry didn't exist then a new variable is created. + * + *	The reference count is decremented for any old value of the variable + *	and incremented for its new value. If the new value for the variable + *	is not the same one referenced by newValuePtr (perhaps as a result of + *	a variable trace), then newValuePtr's ref count is left unchanged by + *	Tcl_SetVar2Ex. newValuePtr's ref count is also left unchanged if we + *	are appending it as a string value: that is, if "flags" includes + *	TCL_APPEND_VALUE but not TCL_LIST_ELEMENT. + * + *	The reference count for the returned object is _not_ incremented: if + *	you want to keep a reference to the object you must increment its ref + *	count yourself. + * + *---------------------------------------------------------------------- + */ + +Tcl_Obj * +Tcl_SetVar2Ex( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be found. */ +    const char *part1,		/* Name of an array (if part2 is non-NULL) or +				 * the name of a variable. */ +    const char *part2,		/* If non-NULL, gives the name of an element +				 * in the array part1. */ +    Tcl_Obj *newValuePtr,	/* New value for variable. */ +    int flags)			/* Various flags that tell how to set value: +				 * any of TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_APPEND_VALUE, TCL_LIST_ELEMENT or +				 * TCL_LEAVE_ERR_MSG. */ +{ +    Tcl_Obj *resPtr, *part2Ptr = NULL, *part1Ptr = Tcl_NewStringObj(part1, -1); + +    Tcl_IncrRefCount(part1Ptr); +    if (part2) { +	part2Ptr = Tcl_NewStringObj(part2, -1); +	Tcl_IncrRefCount(part2Ptr); +    } + +    resPtr = Tcl_ObjSetVar2(interp, part1Ptr, part2Ptr, newValuePtr, flags); +    Tcl_DecrRefCount(part1Ptr); +    if (part2Ptr) { +	Tcl_DecrRefCount(part2Ptr); +    } + +    return resPtr; +} + +/* + *---------------------------------------------------------------------- + * + * Tcl_ObjSetVar2 -- + * + *	This function is the same as Tcl_SetVar2Ex above, except the variable + *	names are passed in Tcl object instead of strings. + * + * Results: + *	Returns a pointer to the Tcl_Obj holding the new value of the + *	variable. If the write operation was disallowed because an array was + *	expected but not found (or vice versa), then NULL is returned; if the + *	TCL_LEAVE_ERR_MSG flag is set, then an explanatory message will be + *	left in the interpreter's result. Note that the returned object may + *	not be the same one referenced by newValuePtr; this is because + *	variable traces may modify the variable's value. + * + * Side effects: + *	The value of the given variable is set. If either the array or the + *	entry didn't exist then a new variable is created. + *	Callers must Incr part1Ptr if they plan to Decr it. + *	Callers must Incr part2Ptr if they plan to Decr it.   *   *----------------------------------------------------------------------   */  Tcl_Obj * -Tcl_ObjSetVar2(interp, part1Ptr, part2Ptr, newValuePtr, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be found. */ -    register Tcl_Obj *part1Ptr;	/* Points to an object holding the name of -				 * an array (if part2 is non-NULL) or the -				 * name of a variable. */ -    register Tcl_Obj *part2Ptr;	/* If non-null, points to an object holding +Tcl_ObjSetVar2( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be found. */ +    register Tcl_Obj *part1Ptr,	/* Points to an object holding the name of an +				 * array (if part2 is non-NULL) or the name of +				 * a variable. */ +    register Tcl_Obj *part2Ptr,	/* If non-NULL, points to an object holding  				 * the name of an element in the array  				 * part1Ptr. */ -    Tcl_Obj *newValuePtr;	/* New value for variable. */ -    int flags;			/* Various flags that tell how to set value: -				 * any of TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY, TCL_APPEND_VALUE, -				 * TCL_LIST_ELEMENT, or TCL_LEAVE_ERR_MSG. */ +    Tcl_Obj *newValuePtr,	/* New value for variable. */ +    int flags)			/* Various flags that tell how to set value: +				 * any of TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_APPEND_VALUE, TCL_LIST_ELEMENT, or +				 * TCL_LEAVE_ERR_MSG. */  { -    char *part1, *part2; +    Var *varPtr, *arrayPtr; -    part1 = Tcl_GetString(part1Ptr); -    if (part2Ptr != NULL) { -	part2 = Tcl_GetString(part2Ptr); -    } else { -	part2 = NULL; +    /* +     * Filter to pass through only the flags this interface supports. +     */ + +    flags &= (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY|TCL_LEAVE_ERR_MSG +	    |TCL_APPEND_VALUE|TCL_LIST_ELEMENT); +    varPtr = TclObjLookupVarEx(interp, part1Ptr, part2Ptr, flags, "set", +	    /*createPart1*/ 1, /*createPart2*/ 1, &arrayPtr); +    if (varPtr == NULL) { +	if (newValuePtr->refCount == 0) { +	    Tcl_DecrRefCount(newValuePtr); +	} +	return NULL;      } -     -    return Tcl_SetVar2Ex(interp, part1, part2, newValuePtr, flags); + +    return TclPtrSetVar(interp, varPtr, arrayPtr, part1Ptr, part2Ptr, +	    newValuePtr, flags, -1);  }  /*   *----------------------------------------------------------------------   * - * Tcl_SetVar2Ex -- + * TclPtrSetVar --   * - *	Given a two-part variable name, which may refer either to a scalar - *	variable or an element of an array, change the value of the variable - *	to a new Tcl object value. If the named scalar or array or element - *	doesn't exist then create one. + *	This function is the same as Tcl_SetVar2Ex above, except that it + *	requires pointers to the variable's Var structs in addition to the + *	variable names.   *   * Results:   *	Returns a pointer to the Tcl_Obj holding the new value of the   *	variable. If the write operation was disallowed because an array was - *	expected but not found (or vice versa), then NULL is returned; if - *	the TCL_LEAVE_ERR_MSG flag is set, then an explanatory message will - *	be left in the interpreter's result. Note that the returned object - *	may not be the same one referenced by newValuePtr; this is because + *	expected but not found (or vice versa), then NULL is returned; if the + *	TCL_LEAVE_ERR_MSG flag is set, then an explanatory message will be + *	left in the interpreter's result. Note that the returned object may + *	not be the same one referenced by newValuePtr; this is because   *	variable traces may modify the variable's value.   *   * Side effects:   *	The value of the given variable is set. If either the array or the   *	entry didn't exist then a new variable is created.   * - *	The reference count is decremented for any old value of the variable - *	and incremented for its new value. If the new value for the variable - *	is not the same one referenced by newValuePtr (perhaps as a result - *	of a variable trace), then newValuePtr's ref count is left unchanged - *	by Tcl_SetVar2Ex. newValuePtr's ref count is also left unchanged if - *	we are appending it as a string value: that is, if "flags" includes - *	TCL_APPEND_VALUE but not TCL_LIST_ELEMENT. - * - *	The reference count for the returned object is _not_ incremented: if - *	you want to keep a reference to the object you must increment its - *	ref count yourself. - *   *----------------------------------------------------------------------   */  Tcl_Obj * -Tcl_SetVar2Ex(interp, part1, part2, newValuePtr, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be found. */ -    char *part1;		/* Name of an array (if part2 is non-NULL) -				 * or the name of a variable. */ -    char *part2;		/* If non-NULL, gives the name of an element +TclPtrSetVar( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be looked up. */ +    register Var *varPtr,	/* Reference to the variable to set. */ +    Var *arrayPtr,		/* Reference to the array containing the +				 * variable, or NULL if the variable is a +				 * scalar. */ +    Tcl_Obj *part1Ptr,		/* Name of an array (if part2 is non-NULL) or +				 * the name of a variable. NULL if the 'index' +				 * parameter is >= 0 */ +    Tcl_Obj *part2Ptr,		/* If non-NULL, gives the name of an element  				 * in the array part1. */ -    Tcl_Obj *newValuePtr;	/* New value for variable. */ -    int flags;			/* Various flags that tell how to set value: -				 * any of TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY, TCL_APPEND_VALUE, -				 * TCL_LIST_ELEMENT or TCL_LEAVE_ERR_MSG. */ +    Tcl_Obj *newValuePtr,	/* New value for variable. */ +    const int flags,		/* OR-ed combination of TCL_GLOBAL_ONLY, and +				 * TCL_LEAVE_ERR_MSG bits. */ +    int index)			/* Index of local var where part1 is to be +				 * found. */  {      Interp *iPtr = (Interp *) interp; -    register Var *varPtr; -    Var *arrayPtr;      Tcl_Obj *oldValuePtr;      Tcl_Obj *resultPtr = NULL;      int result; - -    varPtr = TclLookupVar(interp, part1, part2, flags, "set", -	    /*createPart1*/ 1, /*createPart2*/ 1, &arrayPtr); -    if (varPtr == NULL) { -	return NULL; -    } +    int cleanupOnEarlyError = (newValuePtr->refCount == 0);      /*       * If the variable is in a hashtable and its hPtr field is NULL, then we -     * may have an upvar to an array element where the array was deleted -     * or an upvar to a namespace variable whose namespace was deleted. -     * Generate an error (allowing the variable to be reset would screw up -     * our storage allocation and is meaningless anyway). +     * may have an upvar to an array element where the array was deleted or an +     * upvar to a namespace variable whose namespace was deleted. Generate an +     * error (allowing the variable to be reset would screw up our storage +     * allocation and is meaningless anyway).       */ -    if ((varPtr->flags & VAR_IN_HASHTABLE) && (varPtr->hPtr == NULL)) { +    if (TclIsVarDeadHash(varPtr)) {  	if (flags & TCL_LEAVE_ERR_MSG) {  	    if (TclIsVarArrayElement(varPtr)) { -		VarErrMsg(interp, part1, part2, "set", danglingElement); +		TclObjVarErrMsg(interp, part1Ptr, part2Ptr, "set", +			danglingElement, index); +		Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ELEMENT", NULL);  	    } else { -		VarErrMsg(interp, part1, part2, "set", danglingVar); +		TclObjVarErrMsg(interp, part1Ptr, part2Ptr, "set", +			danglingVar, index); +		Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", NULL);  	    }  	} -	return NULL; +	goto earlyError;      }      /*       * It's an error to try to set an array variable itself.       */ -    if (TclIsVarArray(varPtr) && !TclIsVarUndefined(varPtr)) { +    if (TclIsVarArray(varPtr)) {  	if (flags & TCL_LEAVE_ERR_MSG) { -	    VarErrMsg(interp, part1, part2, "set", isArray); +	    TclObjVarErrMsg(interp, part1Ptr, part2Ptr, "set", isArray,index); +	    Tcl_SetErrorCode(interp, "TCL", "WRITE", "ARRAY", NULL);  	} -	return NULL; +	goto earlyError;      }      /* -     * At this point, if we were appending, we used to call read traces: we -     * treated append as a read-modify-write. However, it seemed unlikely to -     * us that a real program would be interested in such reads being done -     * during a set operation. +     * Invoke any read traces that have been set for the variable if it is +     * requested. This was done for INST_LAPPEND_* but that was inconsistent +     * with the non-bc instruction, and would cause failures trying to +     * lappend to any non-existing ::env var, which is inconsistent with +     * documented behavior. [Bug #3057639].       */ +    if ((flags & TCL_TRACE_READS) && ((varPtr->flags & VAR_TRACED_READ) +	    || (arrayPtr && (arrayPtr->flags & VAR_TRACED_READ)))) { +	if (TCL_ERROR == TclObjCallVarTraces(iPtr, arrayPtr, varPtr, +		part1Ptr, part2Ptr, +		TCL_TRACE_READS, (flags & TCL_LEAVE_ERR_MSG), index)) { +	    goto earlyError; +	} +    } +      /* -     * Set the variable's new value. If appending, append the new value to -     * the variable, either as a list element or as a string. Also, if -     * appending, then if the variable's old value is unshared we can modify -     * it directly, otherwise we must create a new copy to modify: this is -     * "copy on write". +     * Set the variable's new value. If appending, append the new value to the +     * variable, either as a list element or as a string. Also, if appending, +     * then if the variable's old value is unshared we can modify it directly, +     * otherwise we must create a new copy to modify: this is "copy on write".       */      oldValuePtr = varPtr->value.objPtr; -    if (flags & TCL_APPEND_VALUE) { +    if (flags & TCL_LIST_ELEMENT && !(flags & TCL_APPEND_VALUE)) { +	varPtr->value.objPtr = NULL; +    } +    if (flags & (TCL_APPEND_VALUE|TCL_LIST_ELEMENT)) { +#if 0 /* ENABLE_NS_VARNAME_CACHING perhaps? */ +	/* +	 * Can't happen now! +	 */ +  	if (TclIsVarUndefined(varPtr) && (oldValuePtr != NULL)) { -	    Tcl_DecrRefCount(oldValuePtr);     /* discard old value */ +	    TclDecrRefCount(oldValuePtr);	/* Discard old value. */  	    varPtr->value.objPtr = NULL;  	    oldValuePtr = NULL;  	} -	if (flags & TCL_LIST_ELEMENT) {	       /* append list element */ +#endif +	if (flags & TCL_LIST_ELEMENT) {		/* Append list element. */  	    if (oldValuePtr == NULL) {  		TclNewObj(oldValuePtr);  		varPtr->value.objPtr = oldValuePtr; -		Tcl_IncrRefCount(oldValuePtr); /* since var is referenced */ +		Tcl_IncrRefCount(oldValuePtr);	/* Since var is referenced. */  	    } else if (Tcl_IsShared(oldValuePtr)) {  		varPtr->value.objPtr = Tcl_DuplicateObj(oldValuePtr); -		Tcl_DecrRefCount(oldValuePtr); +		TclDecrRefCount(oldValuePtr);  		oldValuePtr = varPtr->value.objPtr; -		Tcl_IncrRefCount(oldValuePtr); /* since var is referenced */ +		Tcl_IncrRefCount(oldValuePtr);	/* Since var is referenced. */  	    }  	    result = Tcl_ListObjAppendElement(interp, oldValuePtr,  		    newValuePtr);  	    if (result != TCL_OK) { -		return NULL; +		goto earlyError;  	    } -	} else {		               /* append string */ +	} else {				/* Append string. */  	    /*  	     * We append newValuePtr's bytes but don't change its ref count.  	     */ @@ -1334,58 +1948,40 @@ Tcl_SetVar2Ex(interp, part1, part2, newValuePtr, flags)  		varPtr->value.objPtr = newValuePtr;  		Tcl_IncrRefCount(newValuePtr);  	    } else { -		if (Tcl_IsShared(oldValuePtr)) {   /* append to copy */ +		if (Tcl_IsShared(oldValuePtr)) {	/* Append to copy. */  		    varPtr->value.objPtr = Tcl_DuplicateObj(oldValuePtr); + +		    TclContinuationsCopy(varPtr->value.objPtr, oldValuePtr); +  		    TclDecrRefCount(oldValuePtr);  		    oldValuePtr = varPtr->value.objPtr; -		    Tcl_IncrRefCount(oldValuePtr); /* since var is ref */ +		    Tcl_IncrRefCount(oldValuePtr);	/* Since var is ref */  		}  		Tcl_AppendObjToObj(oldValuePtr, newValuePtr);  	    }  	}      } else if (newValuePtr != oldValuePtr) {  	/* -	 * In this case we are replacing the value, so we don't need to -	 * do more than swap the objects. +	 * In this case we are replacing the value, so we don't need to do +	 * more than swap the objects.  	 */  	varPtr->value.objPtr = newValuePtr; -	Tcl_IncrRefCount(newValuePtr);      /* var is another ref */ +	Tcl_IncrRefCount(newValuePtr);		/* Var is another ref. */  	if (oldValuePtr != NULL) { -	    TclDecrRefCount(oldValuePtr);   /* discard old value */ +	    TclDecrRefCount(oldValuePtr);	/* Discard old value. */  	}      } -    TclSetVarScalar(varPtr); -    TclClearVarUndefined(varPtr); -    if (arrayPtr != NULL) { -	TclClearVarUndefined(arrayPtr); -    }      /*       * Invoke any write traces for the variable.       */ -    if ((varPtr->tracePtr != NULL) -	    || ((arrayPtr != NULL) && (arrayPtr->tracePtr != NULL))) { -	int resultType; - -	char *msg = CallTraces(iPtr, arrayPtr, varPtr, part1, part2, -	        (flags & (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY)) | TCL_TRACE_WRITES, -		&resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, part1, part2, "set", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, part1, part2, "set", msg); -		} -	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *)msg); -	    } +    if ((varPtr->flags & VAR_TRACED_WRITE) +	    || (arrayPtr && (arrayPtr->flags & VAR_TRACED_WRITE))) { +	if (TCL_ERROR == TclObjCallVarTraces(iPtr, arrayPtr, varPtr, part1Ptr, +		part2Ptr, (flags & (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY)) +		| TCL_TRACE_WRITES, (flags & TCL_LEAVE_ERR_MSG), index)) {  	    goto cleanup;  	}      } @@ -1393,7 +1989,7 @@ Tcl_SetVar2Ex(interp, part1, part2, newValuePtr, flags)      /*       * Return the variable's value unless the variable was changed in some       * gross way by a trace (e.g. it was unset and then recreated as an -     * array).  +     * array).       */      if (TclIsVarScalar(varPtr) && !TclIsVarUndefined(varPtr)) { @@ -1404,1489 +2000,1910 @@ Tcl_SetVar2Ex(interp, part1, part2, newValuePtr, flags)       * A trace changed the value in some gross way. Return an empty string       * object.       */ -     +      resultPtr = iPtr->emptyObjPtr;      /* -     * If the variable doesn't exist anymore and no-one's using it, then -     * free up the relevant structures and hash table entries. +     * If the variable doesn't exist anymore and no-one's using it, then free +     * up the relevant structures and hash table entries.       */ -    cleanup: +  cleanup: +    if (resultPtr == NULL) { +	Tcl_SetErrorCode(interp, "TCL", "WRITE", "VARNAME", NULL); +    }      if (TclIsVarUndefined(varPtr)) { -	CleanupVar(varPtr, arrayPtr); +	TclCleanupVar(varPtr, arrayPtr);      }      return resultPtr; + +  earlyError: +    if (cleanupOnEarlyError) { +	Tcl_DecrRefCount(newValuePtr); +    } +    goto cleanup;  }  /*   *----------------------------------------------------------------------   * - * TclSetIndexedScalar -- + * TclIncrObjVar2 --   * - *	Change the Tcl object value of a local scalar variable in the active - *	procedure, given its compile-time allocated index in the procedure's - *	array of local variables. + *	Given a two-part variable name, which may refer either to a scalar + *	variable or an element of an array, increment the Tcl object value of + *	the variable by a specified Tcl_Obj increment value.   *   * Results:   *	Returns a pointer to the Tcl_Obj holding the new value of the - *	variable given by localIndex. If the specified variable doesn't - *	exist, or there is a clash in array usage, or an error occurs while - *	executing variable traces, then NULL is returned and a message will - *	be left in the interpreter's result if flags has TCL_LEAVE_ERR_MSG. - *	Note that the returned object may not be the same one referenced by - *	newValuePtr; this is because variable traces may modify the - *	variable's value. + *	variable. If the specified variable doesn't exist, or there is a clash + *	in array usage, or an error occurs while executing variable traces, + *	then NULL is returned and a message will be left in the interpreter's + *	result.   *   * Side effects: - *	The value of the given variable is set. The reference count is - *	decremented for any old value of the variable and incremented for - *	its new value. If as a result of a variable trace the new value for - *	the variable is not the same one referenced by newValuePtr, then - *	newValuePtr's ref count is left unchanged. The ref count for the - *	returned object is _not_ incremented to reflect the returned - *	reference; if you want to keep a reference to the object you must - *	increment its ref count yourself. This procedure does not create - *	new variables, but only sets those recognized at compile time. + *	The value of the given variable is incremented by the specified + *	amount. If either the array or the entry didn't exist then a new + *	variable is created. The ref count for the returned object is _not_ + *	incremented to reflect the returned reference; if you want to keep a + *	reference to the object you must increment its ref count yourself. + *	Callers must Incr part1Ptr if they plan to Decr it. + *	Callers must Incr part2Ptr if they plan to Decr it.   *   *----------------------------------------------------------------------   */  Tcl_Obj * -TclSetIndexedScalar(interp, localIndex, newValuePtr, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be found. */ -    int localIndex;		/* Index of variable in procedure's array -				 * of local variables. */ -    Tcl_Obj *newValuePtr;	/* New value for variable. */ -    int flags;			/* Various flags that tell how to set value: -				 * any of TCL_APPEND_VALUE, -				 * TCL_LIST_ELEMENT or TCL_LEAVE_ERR_MSG. */ +TclIncrObjVar2( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be found. */ +    Tcl_Obj *part1Ptr,		/* Points to an object holding the name of an +				 * array (if part2 is non-NULL) or the name of +				 * a variable. */ +    Tcl_Obj *part2Ptr,		/* If non-null, points to an object holding +				 * the name of an element in the array +				 * part1Ptr. */ +    Tcl_Obj *incrPtr,		/* Amount to be added to variable. */ +    int flags)			/* Various flags that tell how to incr value: +				 * any of TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_APPEND_VALUE, TCL_LIST_ELEMENT, +				 * TCL_LEAVE_ERR_MSG. */  { -    Interp *iPtr = (Interp *) interp; -    CallFrame *varFramePtr = iPtr->varFramePtr; -				/* Points to the procedure call frame whose -				 * variables are currently in use. Same as -				 * the current procedure's frame, if any, -				 * unless an "uplevel" is executing. */ -    Var *compiledLocals = varFramePtr->compiledLocals; -    register Var *varPtr;	/* Points to the variable's in-frame Var -				 * structure. */ -    char *varName;		/* Name of the local variable. */ -    Tcl_Obj *oldValuePtr; -    Tcl_Obj *resultPtr = NULL; +    Var *varPtr, *arrayPtr; + +    varPtr = TclObjLookupVarEx(interp, part1Ptr, part2Ptr, flags, "read", +	    1, 1, &arrayPtr); +    if (varPtr == NULL) { +	Tcl_AddErrorInfo(interp, +		"\n    (reading value of variable to increment)"); +	return NULL; +    } +    return TclPtrIncrObjVar(interp, varPtr, arrayPtr, part1Ptr, part2Ptr, +	    incrPtr, flags, -1); +} + +/* + *---------------------------------------------------------------------- + * + * TclPtrIncrObjVar -- + * + *	Given the pointers to a variable and possible containing array, + *	increment the Tcl object value of the variable by a Tcl_Obj increment. + * + * Results: + *	Returns a pointer to the Tcl_Obj holding the new value of the + *	variable. If the specified variable doesn't exist, or there is a clash + *	in array usage, or an error occurs while executing variable traces, + *	then NULL is returned and a message will be left in the interpreter's + *	result. + * + * Side effects: + *	The value of the given variable is incremented by the specified + *	amount. If either the array or the entry didn't exist then a new + *	variable is created. The ref count for the returned object is _not_ + *	incremented to reflect the returned reference; if you want to keep a + *	reference to the object you must increment its ref count yourself. + * + *---------------------------------------------------------------------- + */ -#ifdef TCL_COMPILE_DEBUG -    Proc *procPtr = varFramePtr->procPtr; -    int localCt = procPtr->numCompiledLocals; +Tcl_Obj * +TclPtrIncrObjVar( +    Tcl_Interp *interp,		/* Command interpreter in which variable is to +				 * be found. */ +    Var *varPtr,		/* Reference to the variable to set. */ +    Var *arrayPtr,		/* Reference to the array containing the +				 * variable, or NULL if the variable is a +				 * scalar. */ +    Tcl_Obj *part1Ptr,		/* Points to an object holding the name of an +				 * array (if part2 is non-NULL) or the name of +				 * a variable. */ +    Tcl_Obj *part2Ptr,		/* If non-null, points to an object holding +				 * the name of an element in the array +				 * part1Ptr. */ +    Tcl_Obj *incrPtr,		/* Increment value. */ +/* TODO: Which of these flag values really make sense? */ +    const int flags,		/* Various flags that tell how to incr value: +				 * any of TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_APPEND_VALUE, TCL_LIST_ELEMENT, +				 * TCL_LEAVE_ERR_MSG. */ +    int index)			/* Index into the local variable table of the +				 * variable, or -1. Only used when part1Ptr is +				 * NULL. */ +{ +    register Tcl_Obj *varValuePtr; -    if (compiledLocals == NULL) { -	fprintf(stderr, "\nTclSetIndexedScalar: can't set local %i in frame 0x%x, no compiled locals\n", -		localIndex, (unsigned int) varFramePtr); -	panic("TclSetIndexedScalar: no compiled locals in frame 0x%x", -		(unsigned int) varFramePtr); +    if (TclIsVarInHash(varPtr)) { +	VarHashRefCount(varPtr)++;      } -    if ((localIndex < 0) || (localIndex >= localCt)) { -	fprintf(stderr, "\nTclSetIndexedScalar: can't set local %i in frame 0x%x with %i locals\n", -		localIndex, (unsigned int) varFramePtr, localCt); -	panic("TclSetIndexedScalar: bad local index %i in frame 0x%x", -		localIndex, (unsigned int) varFramePtr); +    varValuePtr = TclPtrGetVar(interp, varPtr, arrayPtr, part1Ptr, part2Ptr, +	    flags, index); +    if (TclIsVarInHash(varPtr)) { +	VarHashRefCount(varPtr)--;      } -#endif /* TCL_COMPILE_DEBUG */ -     -    varPtr = &(compiledLocals[localIndex]); -    varName = varPtr->name; - -    /* -     * If varPtr is a link variable, we have a reference to some variable -     * that was created through an "upvar" or "global" command, or we have a -     * reference to a variable in an enclosing namespace. Traverse through -     * any links until we find the referenced variable. -     */ -	 -    while (TclIsVarLink(varPtr)) { -	varPtr = varPtr->value.linkPtr; +    if (varValuePtr == NULL) { +	varValuePtr = Tcl_NewIntObj(0);      } - -    /* -     * Invoke any read traces that have been set for the variable if we -     * are appending, but only in the lappend case. -     */ - -    if ((flags & TCL_APPEND_VALUE) && (flags & TCL_LIST_ELEMENT) -	    && (varPtr->tracePtr != NULL)) { -	int resultType; - -	char *msg = CallTraces(iPtr, /*arrayPtr*/ NULL, varPtr, varName, NULL, -		TCL_TRACE_READS, &resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, varName, NULL, "read", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, varName, NULL, "read", msg); -		} -	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *)msg); -	    } +    if (Tcl_IsShared(varValuePtr)) { +	/* Copy on write */ +	varValuePtr = Tcl_DuplicateObj(varValuePtr); +	 +	if (TCL_OK == TclIncrObj(interp, varValuePtr, incrPtr)) { +	    return TclPtrSetVar(interp, varPtr, arrayPtr, part1Ptr, part2Ptr, +		    varValuePtr, flags, index); +	} else { +	    Tcl_DecrRefCount(varValuePtr);  	    return NULL;  	} -    } +    } else { +	/* Unshared - can Incr in place */ +	if (TCL_OK == TclIncrObj(interp, varValuePtr, incrPtr)) { -    /* -     * If the variable is in a hashtable and its hPtr field is NULL, then we -     * may have an upvar to an array element where the array was deleted -     * or an upvar to a namespace variable whose namespace was deleted. -     * Generate an error (allowing the variable to be reset would screw up -     * our storage allocation and is meaningless anyway). -     */ +	    /* +	     * This seems dumb to write the incremeted value into the var +	     * after we just adjusted the value in place, but the spec for +	     * [incr] requires that write traces fire, and making this call +	     * is the way to make that happen. +	     */ -    if ((varPtr->flags & VAR_IN_HASHTABLE) && (varPtr->hPtr == NULL)) { -	if (flags & TCL_LEAVE_ERR_MSG) { -	    if (TclIsVarArrayElement(varPtr)) { -		VarErrMsg(interp, varName, NULL, "set", danglingElement); -	    } else { -		VarErrMsg(interp, varName, NULL, "set", danglingVar); -	    } +	    return TclPtrSetVar(interp, varPtr, arrayPtr, part1Ptr, part2Ptr, +		    varValuePtr, flags, index); +	} else { +	    return NULL;  	} -	return NULL;      } +} + +/* + *---------------------------------------------------------------------- + * + * Tcl_UnsetVar -- + * + *	Delete a variable, so that it may not be accessed anymore. + * + * Results: + *	Returns TCL_OK if the variable was successfully deleted, TCL_ERROR if + *	the variable can't be unset. In the event of an error, if the + *	TCL_LEAVE_ERR_MSG flag is set then an error message is left in the + *	interp's result. + * + * Side effects: + *	If varName is defined as a local or global variable in interp, it is + *	deleted. + * + *---------------------------------------------------------------------- + */ -    /* -     * It's an error to try to set an array variable itself. -     */ +#undef Tcl_UnsetVar +int +Tcl_UnsetVar( +    Tcl_Interp *interp,		/* Command interpreter in which varName is to +				 * be looked up. */ +    const char *varName,	/* Name of a variable in interp. May be either +				 * a scalar name or an array name or an +				 * element in an array. */ +    int flags)			/* OR-ed combination of any of +				 * TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY or +				 * TCL_LEAVE_ERR_MSG. */ +{ +    int result; +    Tcl_Obj *varNamePtr; -    if (TclIsVarArray(varPtr) && !TclIsVarUndefined(varPtr)) { -	if (flags & TCL_LEAVE_ERR_MSG) { -	    VarErrMsg(interp, varName, NULL, "set", isArray); -	} -	return NULL; -    } +    varNamePtr = Tcl_NewStringObj(varName, -1); +    Tcl_IncrRefCount(varNamePtr);      /* -     * Set the variable's new value and discard its old value. +     * Filter to pass through only the flags this interface supports.       */ -    oldValuePtr = varPtr->value.objPtr; -    if (flags & TCL_APPEND_VALUE) { -	if (TclIsVarUndefined(varPtr) && (oldValuePtr != NULL)) { -	    Tcl_DecrRefCount(oldValuePtr);     /* discard old value */ -	    varPtr->value.objPtr = NULL; -	    oldValuePtr = NULL; -	} -	if (flags & TCL_LIST_ELEMENT) {	/* append list element */ -	    if (oldValuePtr == NULL) { -		TclNewObj(oldValuePtr); -		varPtr->value.objPtr = oldValuePtr; -		Tcl_IncrRefCount(oldValuePtr); /* since var is referenced */ -	    } else if (Tcl_IsShared(oldValuePtr)) { -		varPtr->value.objPtr = Tcl_DuplicateObj(oldValuePtr); -		Tcl_DecrRefCount(oldValuePtr); -		oldValuePtr = varPtr->value.objPtr; -		Tcl_IncrRefCount(oldValuePtr); /* since var is referenced */ -	    } -	    if (Tcl_ListObjAppendElement(interp, oldValuePtr, -		    newValuePtr) != TCL_OK) { -		return NULL; -	    } -	} else {				/* append string */ -	    /* -	     * We append newValuePtr's bytes but don't change its ref count. -	     */ +    flags &= (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY|TCL_LEAVE_ERR_MSG); +    result = TclObjUnsetVar2(interp, varNamePtr, NULL, flags); -	    if (oldValuePtr == NULL) { -		varPtr->value.objPtr = newValuePtr; -		Tcl_IncrRefCount(newValuePtr); -	    } else { -		if (Tcl_IsShared(oldValuePtr)) {	/* append to copy */ -		    varPtr->value.objPtr = Tcl_DuplicateObj(oldValuePtr); -		    TclDecrRefCount(oldValuePtr); -		    oldValuePtr = varPtr->value.objPtr; -		    Tcl_IncrRefCount(oldValuePtr);	/* since var is ref */ -		} -		Tcl_AppendObjToObj(oldValuePtr, newValuePtr); -	    } -	} -    } else if (newValuePtr != oldValuePtr) {        /* set new value */ -	/* -	 * In this case we are replacing the value, so we don't need to -	 * do more than swap the objects. -	 */ +    Tcl_DecrRefCount(varNamePtr); +    return result; +} + +/* + *---------------------------------------------------------------------- + * + * Tcl_UnsetVar2 -- + * + *	Delete a variable, given a 2-part name. + * + * Results: + *	Returns TCL_OK if the variable was successfully deleted, TCL_ERROR if + *	the variable can't be unset. In the event of an error, if the + *	TCL_LEAVE_ERR_MSG flag is set then an error message is left in the + *	interp's result. + * + * Side effects: + *	If part1 and part2 indicate a local or global variable in interp, it + *	is deleted. If part1 is an array name and part2 is NULL, then the + *	whole array is deleted. + * + *---------------------------------------------------------------------- + */ -	varPtr->value.objPtr = newValuePtr; -	Tcl_IncrRefCount(newValuePtr);       /* var is another ref to obj */ -	if (oldValuePtr != NULL) { -	    TclDecrRefCount(oldValuePtr);    /* discard old value */ -	} +int +Tcl_UnsetVar2( +    Tcl_Interp *interp,		/* Command interpreter in which varName is to +				 * be looked up. */ +    const char *part1,		/* Name of variable or array. */ +    const char *part2,		/* Name of element within array or NULL. */ +    int flags)			/* OR-ed combination of any of +				 * TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_LEAVE_ERR_MSG. */ +{ +    int result; +    Tcl_Obj *part2Ptr = NULL, *part1Ptr = Tcl_NewStringObj(part1, -1); + +    if (part2) { +	part2Ptr = Tcl_NewStringObj(part2, -1);      } -    TclSetVarScalar(varPtr); -    TclClearVarUndefined(varPtr);      /* -     * Invoke any write traces for the variable. +     * Filter to pass through only the flags this interface supports.       */ -    if (varPtr->tracePtr != NULL) { -	int resultType; +    flags &= (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY|TCL_LEAVE_ERR_MSG); +    result = TclObjUnsetVar2(interp, part1Ptr, part2Ptr, flags); -	char *msg = CallTraces(iPtr, /*arrayPtr*/ NULL, varPtr, -	        varName, (char *) NULL, TCL_TRACE_WRITES, &resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, varName, NULL, "set", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, varName, NULL, "set", msg); -		} -	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *)msg); -	    } -	    goto cleanup; -	} +    Tcl_DecrRefCount(part1Ptr); +    if (part2Ptr) { +	Tcl_DecrRefCount(part2Ptr);      } +    return result; +} + +/* + *---------------------------------------------------------------------- + * + * TclObjUnsetVar2 -- + * + *	Delete a variable, given a 2-object name. + * + * Results: + *	Returns TCL_OK if the variable was successfully deleted, TCL_ERROR if + *	the variable can't be unset. In the event of an error, if the + *	TCL_LEAVE_ERR_MSG flag is set then an error message is left in the + *	interp's result. + * + * Side effects: + *	If part1ptr and part2Ptr indicate a local or global variable in + *	interp, it is deleted. If part1Ptr is an array name and part2Ptr is + *	NULL, then the whole array is deleted. + * + *---------------------------------------------------------------------- + */ -    /* -     * Return the variable's value unless the variable was changed in some -     * gross way by a trace (e.g. it was unset and then recreated as an -     * array). If it was changed is a gross way, just return an empty string -     * object. -     */ +int +TclObjUnsetVar2( +    Tcl_Interp *interp,		/* Command interpreter in which varName is to +				 * be looked up. */ +    Tcl_Obj *part1Ptr,		/* Name of variable or array. */ +    Tcl_Obj *part2Ptr,		/* Name of element within array or NULL. */ +    int flags)			/* OR-ed combination of any of +				 * TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_LEAVE_ERR_MSG. */ +{ +    Var *varPtr, *arrayPtr; -    if (TclIsVarScalar(varPtr) && !TclIsVarUndefined(varPtr)) { -	return varPtr->value.objPtr; +    varPtr = TclObjLookupVarEx(interp, part1Ptr, part2Ptr, flags, "unset", +	    /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); +    if (varPtr == NULL) { +	return TCL_ERROR;      } -     -    resultPtr = Tcl_NewObj(); -    /* -     * If the variable doesn't exist anymore and no-one's using it, then -     * free up the relevant structures and hash table entries. -     */ - -    cleanup: -    if (TclIsVarUndefined(varPtr)) { -	CleanupVar(varPtr, NULL); -    } -    return resultPtr; +    return TclPtrUnsetVar(interp, varPtr, arrayPtr, part1Ptr, part2Ptr, flags, +	    -1);  }  /*   *----------------------------------------------------------------------   * - * TclSetElementOfIndexedArray -- + * TclPtrUnsetVar --   * - *	Change the Tcl object value of an element in a local array - *	variable. The element is named by the object elemPtr while the array - *	is specified by its index in the active procedure's array of - *	compiler allocated local variables. + *	Delete a variable, given the pointers to the variable's (and possibly + *	containing array's) VAR structure.   *   * Results: - *	Returns a pointer to the Tcl_Obj holding the new value of the - *	element. If the specified array or element doesn't exist, or there - *	is a clash in array usage, or an error occurs while executing - *	variable traces, then NULL is returned and a message will be left in - *	the interpreter's result if flags has TCL_LEAVE_ERR_MSG. Note that the - *	returned object may not be the same one referenced by newValuePtr; - *	this is because variable traces may modify the variable's value. + *	Returns TCL_OK if the variable was successfully deleted, TCL_ERROR if + *	the variable can't be unset. In the event of an error, if the + *	TCL_LEAVE_ERR_MSG flag is set then an error message is left in the + *	interp's result.   *   * Side effects: - *	The value of the given array element is set. The reference count is - *	decremented for any old value of the element and incremented for its - *	new value. If as a result of a variable trace the new value for the - *	element is not the same one referenced by newValuePtr, then - *	newValuePtr's ref count is left unchanged. The ref count for the - *	returned object is _not_ incremented to reflect the returned - *	reference; if you want to keep a reference to the object you must - *	increment its ref count yourself. This procedure will not create new - *	array variables, but only sets elements of those arrays recognized - *	at compile time. However, if the entry doesn't exist then a new - *	variable is created. + *	If varPtr and arrayPtr indicate a local or global variable in interp, + *	it is deleted. If varPtr is an array reference and part2Ptr is NULL, + *	then the whole array is deleted.   *   *----------------------------------------------------------------------   */ -Tcl_Obj * -TclSetElementOfIndexedArray(interp, localIndex, elemPtr, newValuePtr, flags) -    Tcl_Interp *interp;		/* Command interpreter in which the array is -				 * to be found. */ -    int localIndex;		/* Index of array variable in procedure's -				 * array of local variables. */ -    Tcl_Obj *elemPtr;		/* Points to an object holding the name of -				 * an element to set in the array. */ -    Tcl_Obj *newValuePtr;	/* New value for variable. */ -    int flags;			/* Various flags that tell how to set value: -				 * any of TCL_APPEND_VALUE, -				 * TCL_LIST_ELEMENT or TCL_LEAVE_ERR_MSG. */ +int +TclPtrUnsetVar( +    Tcl_Interp *interp,		/* Command interpreter in which varName is to +				 * be looked up. */ +    register Var *varPtr,	/* The variable to be unset. */ +    Var *arrayPtr,		/* NULL for scalar variables, pointer to the +				 * containing array otherwise. */ +    Tcl_Obj *part1Ptr,		/* Name of an array (if part2 is non-NULL) or +				 * the name of a variable. */ +    Tcl_Obj *part2Ptr,		/* If non-NULL, gives the name of an element +				 * in the array part1. */ +    const int flags,		/* OR-ed combination of any of +				 * TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, +				 * TCL_LEAVE_ERR_MSG. */ +    int index)			/* Index into the local variable table of the +				 * variable, or -1. Only used when part1Ptr is +				 * NULL. */  {      Interp *iPtr = (Interp *) interp; -    CallFrame *varFramePtr = iPtr->varFramePtr; -				/* Points to the procedure call frame whose -				 * variables are currently in use. Same as -				 * the current procedure's frame, if any, -				 * unless an "uplevel" is executing. */ -    Var *compiledLocals = varFramePtr->compiledLocals; -    Var *arrayPtr;		/* Points to the array's in-frame Var -				 * structure. */ -    char *arrayName;		/* Name of the local array. */ -    char *elem; -    Tcl_HashEntry *hPtr; -    Var *varPtr = NULL;		/* Points to the element's Var structure -				 * that we return. */ -    Tcl_Obj *resultPtr = NULL; -    Tcl_Obj *oldValuePtr; -    int new; -     -#ifdef TCL_COMPILE_DEBUG -    Proc *procPtr = varFramePtr->procPtr; -    int localCt = procPtr->numCompiledLocals; - -    if (compiledLocals == NULL) { -	fprintf(stderr, "\nTclSetElementOfIndexedArray: can't set element of local %i in frame 0x%x, no compiled locals\n", -		localIndex, (unsigned int) varFramePtr); -	panic("TclSetIndexedScalar: no compiled locals in frame 0x%x", -		(unsigned int) varFramePtr); -    } -    if ((localIndex < 0) || (localIndex >= localCt)) { -	fprintf(stderr, "\nTclSetIndexedScalar: can't set element of local %i in frame 0x%x with %i locals\n", -		localIndex, (unsigned int) varFramePtr, localCt); -	panic("TclSetElementOfIndexedArray: bad local index %i in frame 0x%x", -		localIndex, (unsigned int) varFramePtr); -    } -#endif /* TCL_COMPILE_DEBUG */ - -    elem = TclGetString(elemPtr); -    arrayPtr = &(compiledLocals[localIndex]); -    arrayName = arrayPtr->name; +    int result = (TclIsVarUndefined(varPtr)? TCL_ERROR : TCL_OK);      /* -     * If arrayPtr is a link variable, we have a reference to some variable -     * that was created through an "upvar" or "global" command, or we have a -     * reference to a variable in an enclosing namespace. Traverse through -     * any links until we find the referenced variable. +     * Keep the variable alive until we're done with it. We used to +     * increase/decrease the refCount for each operation, making it hard to +     * find [Bug 735335] - caused by unsetting the variable whose value was +     * the variable's name.       */ -    while (TclIsVarLink(arrayPtr)) { -	arrayPtr = arrayPtr->value.linkPtr; +    if (TclIsVarInHash(varPtr)) { +	VarHashRefCount(varPtr)++;      } +    UnsetVarStruct(varPtr, arrayPtr, iPtr, part1Ptr, part2Ptr, flags, index); +      /* -     * If the variable is in a hashtable and its hPtr field is NULL, then we -     * may have an upvar to an array element where the array was deleted -     * or an upvar to a namespace variable whose namespace was deleted. -     * Generate an error (allowing the variable to be reset would screw up -     * our storage allocation and is meaningless anyway). +     * It's an error to unset an undefined variable.       */ -    if ((arrayPtr->flags & VAR_IN_HASHTABLE) && (arrayPtr->hPtr == NULL)) { +    if (result != TCL_OK) {  	if (flags & TCL_LEAVE_ERR_MSG) { -	    if (TclIsVarArrayElement(arrayPtr)) { -		VarErrMsg(interp, arrayName, elem, "set", danglingElement); -	    } else { -		VarErrMsg(interp, arrayName, elem, "set", danglingVar); -	    } +	    TclObjVarErrMsg(interp, part1Ptr, part2Ptr, "unset", +		    ((arrayPtr == NULL) ? noSuchVar : noSuchElement), index); +	    Tcl_SetErrorCode(interp, "TCL", "UNSET", "VARNAME", NULL);  	} -	goto errorReturn;      } +#if ENABLE_NS_VARNAME_CACHING      /* -     * Make sure we're dealing with an array. +     * Try to avoid keeping the Var struct allocated due to a tclNsVarNameType +     * keeping a reference. This removes some additional exteriorisations of +     * [Bug 736729], but may be a good thing independently of the bug.       */ -    if (TclIsVarUndefined(arrayPtr) && !TclIsVarArrayElement(arrayPtr)) { -	TclSetVarArray(arrayPtr); -	arrayPtr->value.tablePtr = -	    (Tcl_HashTable *) ckalloc(sizeof(Tcl_HashTable)); -	Tcl_InitHashTable(arrayPtr->value.tablePtr, TCL_STRING_KEYS); -	TclClearVarUndefined(arrayPtr); -    } else if (!TclIsVarArray(arrayPtr)) { -	if (flags & TCL_LEAVE_ERR_MSG) { -	    VarErrMsg(interp, arrayName, elem, "set", needArray); -	} -	goto errorReturn; +    if (part1Ptr->typePtr == &tclNsVarNameType) { +	TclFreeIntRep(part1Ptr);      } +#endif      /* -     * Look up the element. +     * Finally, if the variable is truly not in use then free up its Var +     * structure and remove it from its hash table, if any. The ref count of +     * its value object, if any, was decremented above.       */ -    hPtr = Tcl_CreateHashEntry(arrayPtr->value.tablePtr, elem, &new); -    if (new) { -	if (arrayPtr->searchPtr != NULL) { -	    DeleteSearches(arrayPtr); -	} -	varPtr = NewVar(); -	Tcl_SetHashValue(hPtr, varPtr); -	varPtr->hPtr = hPtr; -        varPtr->nsPtr = varFramePtr->nsPtr; -	TclSetVarArrayElement(varPtr); +    if (TclIsVarInHash(varPtr)) { +	VarHashRefCount(varPtr)--; +	CleanupVar(varPtr, arrayPtr); +    } +    return result; +} + +/* + *---------------------------------------------------------------------- + * + * UnsetVarStruct -- + * + *	Unset and delete a variable. This does the internal work for + *	TclObjUnsetVar2 and TclDeleteNamespaceVars, which call here for each + *	variable to be unset and deleted. + * + * Results: + *	None. + * + * Side effects: + *	If the arguments indicate a local or global variable in iPtr, it is + *	unset and deleted. + * + *---------------------------------------------------------------------- + */ + +static void +UnsetVarStruct( +    Var *varPtr, +    Var *arrayPtr, +    Interp *iPtr, +    Tcl_Obj *part1Ptr, +    Tcl_Obj *part2Ptr, +    int flags, +    int index) +{ +    Var dummyVar; +    int traced = TclIsVarTraced(varPtr) +	    || (arrayPtr && (arrayPtr->flags & VAR_TRACED_UNSET)); + +    if (arrayPtr && (arrayPtr->flags & VAR_SEARCH_ACTIVE)) { +	DeleteSearches(iPtr, arrayPtr); +    } else if (varPtr->flags & VAR_SEARCH_ACTIVE) { +	DeleteSearches(iPtr, varPtr);      } -    varPtr = (Var *) Tcl_GetHashValue(hPtr);      /* -     * It's an error to try to set an array variable itself. +     * The code below is tricky, because of the possibility that a trace +     * function might try to access a variable being deleted. To handle this +     * situation gracefully, do things in three steps: +     * 1. Copy the contents of the variable to a dummy variable structure, and +     *    mark the original Var structure as undefined. +     * 2. Invoke traces and clean up the variable, using the dummy copy. +     * 3. If at the end of this the original variable is still undefined and +     *    has no outstanding references, then delete it (but it could have +     *    gotten recreated by a trace).       */ -    if (TclIsVarArray(varPtr)) { -	if (flags & TCL_LEAVE_ERR_MSG) { -	    VarErrMsg(interp, arrayName, elem, "set", isArray); -	} -	goto errorReturn; -    } +    dummyVar = *varPtr; +    dummyVar.flags &= ~VAR_ALL_HASH; +    TclSetVarUndefined(varPtr);      /* -     * Invoke any read traces that have been set for the element variable if -     * we are appending, but only in the lappend case. +     * Call trace functions for the variable being deleted. Then delete its +     * traces. Be sure to abort any other traces for the variable that are +     * still pending. Special tricks: +     * 1. We need to increment varPtr's refCount around this: TclCallVarTraces +     *    will use dummyVar so it won't increment varPtr's refCount itself. +     * 2. Turn off the VAR_TRACE_ACTIVE flag in dummyVar: we want to call +     *    unset traces even if other traces are pending.       */ -    if ((flags & TCL_APPEND_VALUE) && (flags & TCL_LIST_ELEMENT) -	    && ((varPtr->tracePtr != NULL) -		    || ((arrayPtr != NULL) && (arrayPtr->tracePtr != NULL)))) { -	int resultType; +    if (traced) { +	VarTrace *tracePtr = NULL; +	Tcl_HashEntry *tPtr; -	char *msg = CallTraces(iPtr, arrayPtr, varPtr, arrayName, elem, -		TCL_TRACE_READS, &resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, arrayName, elem, "read", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, arrayName, elem, "read", msg); -		} -	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *) msg); +	if (TclIsVarTraced(&dummyVar)) { +	    /* +	     * Transfer any existing traces on var, IF there are unset traces. +	     * Otherwise just delete them. +	     */ + +	    int isNew; + +	    tPtr = Tcl_FindHashEntry(&iPtr->varTraces, varPtr); +	    tracePtr = Tcl_GetHashValue(tPtr); +	    varPtr->flags &= ~VAR_ALL_TRACES; +	    Tcl_DeleteHashEntry(tPtr); +	    if (dummyVar.flags & VAR_TRACED_UNSET) { +		tPtr = Tcl_CreateHashEntry(&iPtr->varTraces, +			&dummyVar, &isNew); +		Tcl_SetHashValue(tPtr, tracePtr);  	    } -	    goto errorReturn;  	} -    } -    /* -     * Set the variable's new value and discard the old one. -     */ +	if ((dummyVar.flags & VAR_TRACED_UNSET) +		|| (arrayPtr && (arrayPtr->flags & VAR_TRACED_UNSET))) { +	    dummyVar.flags &= ~VAR_TRACE_ACTIVE; +	    TclObjCallVarTraces(iPtr, arrayPtr, &dummyVar, part1Ptr, part2Ptr, +		    (flags & (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY)) +			    | TCL_TRACE_UNSETS, +		    /* leaveErrMsg */ 0, index); -    oldValuePtr = varPtr->value.objPtr; -    if (flags & TCL_APPEND_VALUE) { -	if (TclIsVarUndefined(varPtr) && (oldValuePtr != NULL)) { -	    Tcl_DecrRefCount(oldValuePtr);     /* discard old value */ -	    varPtr->value.objPtr = NULL; -	    oldValuePtr = NULL; -	} -	if (flags & TCL_LIST_ELEMENT) {	/* append list element */ -	    if (oldValuePtr == NULL) { -		TclNewObj(oldValuePtr); -		varPtr->value.objPtr = oldValuePtr; -		Tcl_IncrRefCount(oldValuePtr); /* since var is referenced */ -	    } else if (Tcl_IsShared(oldValuePtr)) { -		varPtr->value.objPtr = Tcl_DuplicateObj(oldValuePtr); -		Tcl_DecrRefCount(oldValuePtr); -		oldValuePtr = varPtr->value.objPtr; -		Tcl_IncrRefCount(oldValuePtr); /* since var is referenced */ -	    } -	    if (Tcl_ListObjAppendElement(interp, oldValuePtr, -		    newValuePtr) != TCL_OK) { -		return NULL; -	    } -	} else {				/* append string */  	    /* -	     * We append newValuePtr's bytes but don't change its ref count. +	     * The traces that we just called may have triggered a change in +	     * the set of traces. If so, reload the traces to manipulate.  	     */ -	    if (oldValuePtr == NULL) { -		varPtr->value.objPtr = newValuePtr; -		Tcl_IncrRefCount(newValuePtr); -	    } else { -		if (Tcl_IsShared(oldValuePtr)) {	/* append to copy */ -		    varPtr->value.objPtr = Tcl_DuplicateObj(oldValuePtr); -		    TclDecrRefCount(oldValuePtr); -		    oldValuePtr = varPtr->value.objPtr; -		    Tcl_IncrRefCount(oldValuePtr);	/* since var is ref */ +	    tracePtr = NULL; +	    if (TclIsVarTraced(&dummyVar)) { +		tPtr = Tcl_FindHashEntry(&iPtr->varTraces, &dummyVar); +		tracePtr = Tcl_GetHashValue(tPtr); +		if (tPtr) { +		    Tcl_DeleteHashEntry(tPtr);  		} -		Tcl_AppendObjToObj(oldValuePtr, newValuePtr);  	    }  	} -    } else if (newValuePtr != oldValuePtr) {	/* set new value */ -	/* -	 * In this case we are replacing the value, so we don't need to -	 * do more than swap the objects. -	 */ - -	varPtr->value.objPtr = newValuePtr; -	Tcl_IncrRefCount(newValuePtr);		/* var is another ref to obj */ -	if (oldValuePtr != NULL) { -	    TclDecrRefCount(oldValuePtr);	/* discard old value */ -	} -    } -    TclSetVarScalar(varPtr); -    TclClearVarUndefined(varPtr); -    /* -     * Invoke any write traces for the element variable. -     */ +	if (tracePtr) { +	    ActiveVarTrace *activePtr; -    if ((varPtr->tracePtr != NULL) -	    || ((arrayPtr != NULL) && (arrayPtr->tracePtr != NULL))) { -	int resultType; +	    while (tracePtr) { +		VarTrace *prevPtr = tracePtr; -	char *msg = CallTraces(iPtr, arrayPtr, varPtr, arrayName, elem, -		TCL_TRACE_WRITES, &resultType); -	if (msg != NULL) { -	    if (flags & TCL_LEAVE_ERR_MSG) { -		if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    VarErrMsg(interp, arrayName, elem, "set", -			      Tcl_GetString((Tcl_Obj *) msg)); -		} else { -		    VarErrMsg(interp, arrayName, elem, "set", msg); -		} +		tracePtr = tracePtr->nextPtr; +		prevPtr->nextPtr = NULL; +		Tcl_EventuallyFree(prevPtr, TCL_DYNAMIC);  	    } -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *) msg); +	    for (activePtr = iPtr->activeVarTracePtr;  activePtr != NULL; +		    activePtr = activePtr->nextPtr) { +		if (activePtr->varPtr == varPtr) { +		    activePtr->nextTracePtr = NULL; +		}  	    } -	    goto errorReturn; +	    dummyVar.flags &= ~VAR_ALL_TRACES;  	}      } -    /* -     * Return the element's value unless it was changed in some gross way by -     * a trace (e.g. it was unset and then recreated as an array). If it was -     * changed is a gross way, just return an empty string object. -     */ +    if (TclIsVarScalar(&dummyVar) && (dummyVar.value.objPtr != NULL)) { +	/* +	 * Decrement the ref count of the var's value. +	 */ -    if (TclIsVarScalar(varPtr) && !TclIsVarUndefined(varPtr)) { -	return varPtr->value.objPtr; +	Tcl_Obj *objPtr = dummyVar.value.objPtr; + +	TclDecrRefCount(objPtr); +    } else if (TclIsVarArray(&dummyVar)) { +	/* +	 * If the variable is an array, delete all of its elements. This must +	 * be done after calling and deleting the traces on the array, above +	 * (that's the way traces are defined). If the array name is not +	 * present and is required for a trace on some element, it will be +	 * computed at DeleteArray. +	 */ + +	DeleteArray(iPtr, part1Ptr, (Var *) &dummyVar, (flags +		& (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY)) | TCL_TRACE_UNSETS, +		index); +    } else if (TclIsVarLink(&dummyVar)) { +	/* +	 * For global/upvar variables referenced in procedures, decrement the +	 * reference count on the variable referred to, and free the +	 * referenced variable if it's no longer needed. +	 */ + +	Var *linkPtr = dummyVar.value.linkPtr; + +	if (TclIsVarInHash(linkPtr)) { +	    VarHashRefCount(linkPtr)--; +	    CleanupVar(linkPtr, NULL); +	}      } -     -    resultPtr = Tcl_NewObj();      /* -     * An error. If the variable doesn't exist anymore and no-one's using -     * it, then free up the relevant structures and hash table entries. +     * If the variable was a namespace variable, decrement its reference +     * count.       */ -    errorReturn: -    if ((varPtr != NULL) && TclIsVarUndefined(varPtr)) { -	CleanupVar(varPtr, NULL); /* note: array isn't in hashtable */ -    } -    return resultPtr; +    TclClearVarNamespaceVar(varPtr);  }  /*   *----------------------------------------------------------------------   * - * TclIncrVar2 -- + * Tcl_UnsetObjCmd --   * - *	Given a two-part variable name, which may refer either to a scalar - *	variable or an element of an array, increment the Tcl object value - *	of the variable by a specified amount. + *	This object-based function is invoked to process the "unset" Tcl + *	command. See the user documentation for details on what it does.   *   * Results: - *	Returns a pointer to the Tcl_Obj holding the new value of the - *	variable. If the specified variable doesn't exist, or there is a - *	clash in array usage, or an error occurs while executing variable - *	traces, then NULL is returned and a message will be left in - *	the interpreter's result. + *	A standard Tcl object result value.   *   * Side effects: - *	The value of the given variable is incremented by the specified - *	amount. If either the array or the entry didn't exist then a new - *	variable is created. The ref count for the returned object is _not_ - *	incremented to reflect the returned reference; if you want to keep a - *	reference to the object you must increment its ref count yourself. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -Tcl_Obj * -TclIncrVar2(interp, part1Ptr, part2Ptr, incrAmount, flags) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be found. */ -    Tcl_Obj *part1Ptr;		/* Points to an object holding the name of -				 * an array (if part2 is non-NULL) or the -				 * name of a variable. */ -    Tcl_Obj *part2Ptr;		/* If non-null, points to an object holding -				 * the name of an element in the array -				 * part1Ptr. */ -    long incrAmount;		/* Amount to be added to variable. */ -    int flags;                  /* Various flags that tell how to incr value: -				 * any of TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY, TCL_APPEND_VALUE, -				 * TCL_LIST_ELEMENT, TCL_LEAVE_ERR_MSG. */ +	/* ARGSUSED */ +int +Tcl_UnsetObjCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  { -    register Tcl_Obj *varValuePtr; -    int createdNewObj;		/* Set 1 if var's value object is shared -				 * so we must increment a copy (i.e. copy -				 * on write). */ -    long i; +    register int i, flags = TCL_LEAVE_ERR_MSG; +    register const char *name; -    varValuePtr = Tcl_ObjGetVar2(interp, part1Ptr, part2Ptr, flags); -    if (varValuePtr == NULL) { -	Tcl_AddObjErrorInfo(interp, -		"\n    (reading value of variable to increment)", -1); -	return NULL; +    if (objc == 1) { +	/* +	 * Do nothing if no arguments supplied, so as to match command +	 * documentation. +	 */ + +	return TCL_OK;      }      /* -     * Increment the variable's value. If the object is unshared we can -     * modify it directly, otherwise we must create a new copy to modify: -     * this is "copy on write". Then free the variable's old string -     * representation, if any, since it will no longer be valid. +     * Simple, restrictive argument parsing. The only options are -- and +     * -nocomplain (which must come first and be given exactly to be an +     * option).       */ -    createdNewObj = 0; -    if (Tcl_IsShared(varValuePtr)) { -	varValuePtr = Tcl_DuplicateObj(varValuePtr); -	createdNewObj = 1; -    } -#ifdef TCL_WIDE_INT_IS_LONG -    if (Tcl_GetLongFromObj(interp, varValuePtr, &i) != TCL_OK) { -	if (createdNewObj) { -	    Tcl_DecrRefCount(varValuePtr); /* free unneeded copy */ -	} -	return NULL; -    } -    Tcl_SetLongObj(varValuePtr, (i + incrAmount)); -#else -    if (varValuePtr->typePtr == &tclWideIntType) { -	Tcl_WideInt wide = varValuePtr->internalRep.wideValue; -	Tcl_SetWideIntObj(varValuePtr, wide + Tcl_LongAsWide(incrAmount)); -    } else if (varValuePtr->typePtr == &tclIntType) { -	i = varValuePtr->internalRep.longValue; -	Tcl_SetIntObj(varValuePtr, i + incrAmount); -    } else { -	/* -	 * Not an integer or wide internal-rep... -	 */ -	Tcl_WideInt wide; -	if (Tcl_GetWideIntFromObj(interp, varValuePtr, &wide) != TCL_OK) { -	    if (createdNewObj) { -		Tcl_DecrRefCount(varValuePtr); /* free unneeded copy */ +    i = 1; +    name = TclGetString(objv[i]); +    if (name[0] == '-') { +	if (strcmp("-nocomplain", name) == 0) { +	    i++; +	    if (i == objc) { +		return TCL_OK;  	    } -	    return NULL; +	    flags = 0; +	    name = TclGetString(objv[i]);  	} -	if (wide <= Tcl_LongAsWide(LONG_MAX) -		&& wide >= Tcl_LongAsWide(LONG_MIN)) { -	    Tcl_SetLongObj(varValuePtr, Tcl_WideAsLong(wide) + incrAmount); -	} else { -	    Tcl_SetWideIntObj(varValuePtr, wide + Tcl_LongAsWide(incrAmount)); +	if (strcmp("--", name) == 0) { +	    i++;  	}      } -#endif -    /* -     * Store the variable's new value and run any write traces. -     */ -     -    return Tcl_ObjSetVar2(interp, part1Ptr, part2Ptr, varValuePtr, flags); +    for (; i < objc; i++) { +	if ((TclObjUnsetVar2(interp, objv[i], NULL, flags) != TCL_OK) +		&& (flags == TCL_LEAVE_ERR_MSG)) { +	    return TCL_ERROR; +	} +    } +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * TclIncrIndexedScalar -- + * Tcl_AppendObjCmd --   * - *	Increments the Tcl object value of a local scalar variable in the - *	active procedure, given its compile-time allocated index in the - *	procedure's array of local variables. + *	This object-based function is invoked to process the "append" Tcl + *	command. See the user documentation for details on what it does.   *   * Results: - *	Returns a pointer to the Tcl_Obj holding the new value of the - *	variable given by localIndex. If the specified variable doesn't - *	exist, or there is a clash in array usage, or an error occurs while - *	executing variable traces, then NULL is returned and a message will - *	be left in the interpreter's result.  + *	A standard Tcl object result value.   *   * Side effects: - *	The value of the given variable is incremented by the specified - *	amount. The ref count for the returned object is _not_ incremented - *	to reflect the returned reference; if you want to keep a reference - *	to the object you must increment its ref count yourself. + *	A variable's value may be changed.   *   *----------------------------------------------------------------------   */ -Tcl_Obj * -TclIncrIndexedScalar(interp, localIndex, incrAmount) -    Tcl_Interp *interp;		/* Command interpreter in which variable is -				 * to be found. */ -    int localIndex;		/* Index of variable in procedure's array -				 * of local variables. */ -    long incrAmount;		/* Amount to be added to variable. */ +	/* ARGSUSED */ +int +Tcl_AppendObjCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  { -    register Tcl_Obj *varValuePtr; -    int createdNewObj;		/* Set 1 if var's value object is shared -				 * so we must increment a copy (i.e. copy -				 * on write). */ -    long i; +    Var *varPtr, *arrayPtr; +    register Tcl_Obj *varValuePtr = NULL; +				/* Initialized to avoid compiler warning. */ +    int i; -    varValuePtr = TclGetIndexedScalar(interp, localIndex, TCL_LEAVE_ERR_MSG); -    if (varValuePtr == NULL) { -	Tcl_AddObjErrorInfo(interp, -		"\n    (reading value of variable to increment)", -1); -	return NULL; +    if (objc < 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "varName ?value ...?"); +	return TCL_ERROR;      } -    /* -     * Reach into the object's representation to extract and increment the -     * variable's value. If the object is unshared we can modify it -     * directly, otherwise we must create a new copy to modify: this is -     * "copy on write". Then free the variable's old string representation, -     * if any, since it will no longer be valid. -     */ +    if (objc == 2) { +	varValuePtr = Tcl_ObjGetVar2(interp, objv[1], NULL,TCL_LEAVE_ERR_MSG); +	if (varValuePtr == NULL) { +	    return TCL_ERROR; +	} +    } else { +	varPtr = TclObjLookupVarEx(interp, objv[1], NULL, TCL_LEAVE_ERR_MSG, +		"set", /*createPart1*/ 1, /*createPart2*/ 1, &arrayPtr); +	if (varPtr == NULL) { +	    return TCL_ERROR; +	} +	for (i=2 ; i<objc ; i++) { +	    /* +	     * Note that we do not need to increase the refCount of the Var +	     * pointers: should a trace delete the variable, the return value +	     * of TclPtrSetVar will be NULL or emptyObjPtr, and we will not +	     * access the variable again. +	     */ -    createdNewObj = 0; -    if (Tcl_IsShared(varValuePtr)) { -	createdNewObj = 1; -	varValuePtr = Tcl_DuplicateObj(varValuePtr); -    } -#ifdef TCL_WIDE_INT_IS_LONG -    if (Tcl_GetLongFromObj(interp, varValuePtr, &i) != TCL_OK) { -	if (createdNewObj) { -	    Tcl_DecrRefCount(varValuePtr); /* free unneeded copy */ +	    varValuePtr = TclPtrSetVar(interp, varPtr, arrayPtr, objv[1], +		    NULL, objv[i], TCL_APPEND_VALUE|TCL_LEAVE_ERR_MSG, -1); +	    if ((varValuePtr == NULL) || +		    (varValuePtr == ((Interp *) interp)->emptyObjPtr)) { +		return TCL_ERROR; +	    }  	} -	return NULL;      } -    Tcl_SetLongObj(varValuePtr, (i + incrAmount)); -#else -    if (varValuePtr->typePtr == &tclWideIntType) { -	Tcl_WideInt wide = varValuePtr->internalRep.wideValue; -	Tcl_SetWideIntObj(varValuePtr, wide + Tcl_LongAsWide(incrAmount)); -    } else if (varValuePtr->typePtr == &tclIntType) { -	i = varValuePtr->internalRep.longValue; -	Tcl_SetIntObj(varValuePtr, i + incrAmount); +    Tcl_SetObjResult(interp, varValuePtr); +    return TCL_OK; +} + +/* + *---------------------------------------------------------------------- + * + * Tcl_LappendObjCmd -- + * + *	This object-based function is invoked to process the "lappend" Tcl + *	command. See the user documentation for details on what it does. + * + * Results: + *	A standard Tcl object result value. + * + * Side effects: + *	A variable's value may be changed. + * + *---------------------------------------------------------------------- + */ + +	/* ARGSUSED */ +int +Tcl_LappendObjCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */ +{ +    Tcl_Obj *varValuePtr, *newValuePtr; +    int numElems, createdNewObj; +    Var *varPtr, *arrayPtr; +    int result; + +    if (objc < 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "varName ?value ...?"); +	return TCL_ERROR; +    } +    if (objc == 2) { +	newValuePtr = Tcl_ObjGetVar2(interp, objv[1], NULL, 0); +	if (newValuePtr == NULL) { +	    /* +	     * The variable doesn't exist yet. Just create it with an empty +	     * initial value. +	     */ + +	    TclNewObj(varValuePtr); +	    newValuePtr = Tcl_ObjSetVar2(interp, objv[1], NULL, varValuePtr, +		    TCL_LEAVE_ERR_MSG); +	    if (newValuePtr == NULL) { +		return TCL_ERROR; +	    } +	} else { +	    result = TclListObjLength(interp, newValuePtr, &numElems); +	    if (result != TCL_OK) { +		return result; +	    } +	}      } else {  	/* -	 * Not an integer or wide internal-rep... +	 * We have arguments to append. We used to call Tcl_SetVar2 to append +	 * each argument one at a time to ensure that traces were run for each +	 * append step. We now append the arguments all at once because it's +	 * faster. Note that a read trace and a write trace for the variable +	 * will now each only be called once. Also, if the variable's old +	 * value is unshared we modify it directly, otherwise we create a new +	 * copy to modify: this is "copy on write". +	 */ + +	createdNewObj = 0; + +	/* +	 * Protect the variable pointers around the TclPtrGetVar call +	 * to insure that they remain valid even if the variable was undefined +	 * and unused.  	 */ -	Tcl_WideInt wide; -	if (Tcl_GetWideIntFromObj(interp, varValuePtr, &wide) != TCL_OK) { + +	varPtr = TclObjLookupVarEx(interp, objv[1], NULL, TCL_LEAVE_ERR_MSG, +		"set", /*createPart1*/ 1, /*createPart2*/ 1, &arrayPtr); +	if (varPtr == NULL) { +	    return TCL_ERROR; +	} +	if (TclIsVarInHash(varPtr)) { +	    VarHashRefCount(varPtr)++; +	} +	if (arrayPtr && TclIsVarInHash(arrayPtr)) { +	    VarHashRefCount(arrayPtr)++; +	} +	varValuePtr = TclPtrGetVar(interp, varPtr, arrayPtr, objv[1], NULL, +		TCL_LEAVE_ERR_MSG, -1); +	if (TclIsVarInHash(varPtr)) { +	    VarHashRefCount(varPtr)--; +	} +	if (arrayPtr && TclIsVarInHash(arrayPtr)) { +	    VarHashRefCount(arrayPtr)--; +	} + +	if (varValuePtr == NULL) { +	    /* +	     * We couldn't read the old value: either the var doesn't yet +	     * exist or it's an array element. If it's new, we will try to +	     * create it with Tcl_ObjSetVar2 below. +	     */ + +	    TclNewObj(varValuePtr); +	    createdNewObj = 1; +	} else if (Tcl_IsShared(varValuePtr)) { +	    varValuePtr = Tcl_DuplicateObj(varValuePtr); +	    createdNewObj = 1; +	} + +	result = TclListObjLength(interp, varValuePtr, &numElems); +	if (result == TCL_OK) { +	    result = Tcl_ListObjReplace(interp, varValuePtr, numElems, 0, +		    (objc-2), (objv+2)); +	} +	if (result != TCL_OK) {  	    if (createdNewObj) { -		Tcl_DecrRefCount(varValuePtr); /* free unneeded copy */ +		TclDecrRefCount(varValuePtr); /* Free unneeded obj. */  	    } -	    return NULL; +	    return result;  	} -	if (wide <= Tcl_LongAsWide(LONG_MAX) -		&& wide >= Tcl_LongAsWide(LONG_MIN)) { -	    Tcl_SetLongObj(varValuePtr, Tcl_WideAsLong(wide) + incrAmount); -	} else { -	    Tcl_SetWideIntObj(varValuePtr, wide + Tcl_LongAsWide(incrAmount)); + +	/* +	 * Now store the list object back into the variable. If there is an +	 * error setting the new value, decrement its ref count if it was new +	 * and we didn't create the variable. +	 */ + +	newValuePtr = TclPtrSetVar(interp, varPtr, arrayPtr, objv[1], NULL, +		varValuePtr, TCL_LEAVE_ERR_MSG, -1); +	if (newValuePtr == NULL) { +	    return TCL_ERROR;  	}      } -#endif      /* -     * Store the variable's new value and run any write traces. +     * Set the interpreter's object result to refer to the variable's value +     * object.       */ -     -    return TclSetIndexedScalar(interp, localIndex, varValuePtr, -	    TCL_LEAVE_ERR_MSG); + +    Tcl_SetObjResult(interp, newValuePtr); +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * TclIncrElementOfIndexedArray -- + * TclArraySet --   * - *	Increments the Tcl object value of an element in a local array - *	variable. The element is named by the object elemPtr while the array - *	is specified by its index in the active procedure's array of - *	compiler allocated local variables. + *	Set the elements of an array. If there are no elements to set, create + *	an empty array. This routine is used by the Tcl_ArrayObjCmd and by the + *	TclSetupEnv routine.   *   * Results: - *	Returns a pointer to the Tcl_Obj holding the new value of the - *	element. If the specified array or element doesn't exist, or there - *	is a clash in array usage, or an error occurs while executing - *	variable traces, then NULL is returned and a message will be left in - *	the interpreter's result. + *	A standard Tcl result object.   *   * Side effects: - *	The value of the given array element is incremented by the specified - *	amount. The ref count for the returned object is _not_ incremented - *	to reflect the returned reference; if you want to keep a reference - *	to the object you must increment its ref count yourself. If the - *	entry doesn't exist then a new variable is created. + *	A variable will be created if one does not already exist. + *	Callers must Incr arrayNameObj if they pland to Decr it.   *   *----------------------------------------------------------------------   */ -Tcl_Obj * -TclIncrElementOfIndexedArray(interp, localIndex, elemPtr, incrAmount) -    Tcl_Interp *interp;		/* Command interpreter in which the array is -				 * to be found. */ -    int localIndex;		/* Index of array variable in procedure's -				 * array of local variables. */ -    Tcl_Obj *elemPtr;		/* Points to an object holding the name of -				 * an element to increment in the array. */ -    long incrAmount;		/* Amount to be added to variable. */ +int +TclArraySet( +    Tcl_Interp *interp,		/* Current interpreter. */ +    Tcl_Obj *arrayNameObj,	/* The array name. */ +    Tcl_Obj *arrayElemObj)	/* The array elements list or dict. If this is +				 * NULL, create an empty array. */  { -    register Tcl_Obj *varValuePtr; -    int createdNewObj;		/* Set 1 if var's value object is shared -				 * so we must increment a copy (i.e. copy -				 * on write). */ -    long i; +    Var *varPtr, *arrayPtr; +    int result, i; -    varValuePtr = TclGetElementOfIndexedArray(interp, localIndex, elemPtr, -	    TCL_LEAVE_ERR_MSG); -    if (varValuePtr == NULL) { -	Tcl_AddObjErrorInfo(interp, -		"\n    (reading value of variable to increment)", -1); -	return NULL; +    varPtr = TclObjLookupVarEx(interp, arrayNameObj, NULL, +	    /*flags*/ TCL_LEAVE_ERR_MSG, /*msg*/ "set", /*createPart1*/ 1, +	    /*createPart2*/ 1, &arrayPtr); +    if (varPtr == NULL) { +	return TCL_ERROR; +    } +    if (arrayPtr) { +	CleanupVar(varPtr, arrayPtr); +	TclObjVarErrMsg(interp, arrayNameObj, NULL, "set", needArray, -1); +	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", +		TclGetString(arrayNameObj), NULL); +	return TCL_ERROR; +    } + +    if (arrayElemObj == NULL) { +	goto ensureArray;      }      /* -     * Reach into the object's representation to extract and increment the -     * variable's value. If the object is unshared we can modify it -     * directly, otherwise we must create a new copy to modify: this is -     * "copy on write". Then free the variable's old string representation, -     * if any, since it will no longer be valid. +     * Install the contents of the dictionary or list into the array.       */ -    createdNewObj = 0; -    if (Tcl_IsShared(varValuePtr)) { -	createdNewObj = 1; -	varValuePtr = Tcl_DuplicateObj(varValuePtr); -    } -#ifdef TCL_WIDE_INT_IS_LONG -    if (Tcl_GetLongFromObj(interp, varValuePtr, &i) != TCL_OK) { -	if (createdNewObj) { -	    Tcl_DecrRefCount(varValuePtr); /* free unneeded copy */ +    if (arrayElemObj->typePtr == &tclDictType) { +	Tcl_Obj *keyPtr, *valuePtr; +	Tcl_DictSearch search; +	int done; + +	if (Tcl_DictObjSize(interp, arrayElemObj, &done) != TCL_OK) { +	    return TCL_ERROR;  	} -	return NULL; -    } -    Tcl_SetLongObj(varValuePtr, (i + incrAmount)); -#else -    if (varValuePtr->typePtr == &tclWideIntType) { -	Tcl_WideInt wide = varValuePtr->internalRep.wideValue; -	Tcl_SetWideIntObj(varValuePtr, wide + Tcl_LongAsWide(incrAmount)); -    } else if (varValuePtr->typePtr == &tclIntType) { -	i = varValuePtr->internalRep.longValue; -	Tcl_SetIntObj(varValuePtr, i + incrAmount); -    } else { +	if (done == 0) { +	    /* +	     * Empty, so we'll just force the array to be properly existing +	     * instead. +	     */ + +	    goto ensureArray; +	} +  	/* -	 * Not an integer or wide internal-rep... +	 * Don't need to look at result of Tcl_DictObjFirst as we've just +	 * successfully used a dictionary operation on the same object.  	 */ -	Tcl_WideInt wide; -	if (Tcl_GetWideIntFromObj(interp, varValuePtr, &wide) != TCL_OK) { -	    if (createdNewObj) { -		Tcl_DecrRefCount(varValuePtr); /* free unneeded copy */ + +	for (Tcl_DictObjFirst(interp, arrayElemObj, &search, +		&keyPtr, &valuePtr, &done) ; !done ; +		Tcl_DictObjNext(&search, &keyPtr, &valuePtr, &done)) { +	    /* +	     * At this point, it would be nice if the key was directly usable +	     * by the array. This isn't the case though. +	     */ + +	    Var *elemVarPtr = TclLookupArrayElement(interp, arrayNameObj, +		    keyPtr, TCL_LEAVE_ERR_MSG, "set", 1, 1, varPtr, -1); + +	    if ((elemVarPtr == NULL) || +		    (TclPtrSetVar(interp, elemVarPtr, varPtr, arrayNameObj, +		    keyPtr, valuePtr, TCL_LEAVE_ERR_MSG, -1) == NULL)) { +		Tcl_DictObjDone(&search); +		return TCL_ERROR;  	    } -	    return NULL;  	} -	if (wide <= Tcl_LongAsWide(LONG_MAX) -		&& wide >= Tcl_LongAsWide(LONG_MIN)) { -	    Tcl_SetLongObj(varValuePtr, Tcl_WideAsLong(wide) + incrAmount); -	} else { -	    Tcl_SetWideIntObj(varValuePtr, wide + Tcl_LongAsWide(incrAmount)); +	return TCL_OK; +    } else { +	/* +	 * Not a dictionary, so assume (and convert to, for backward- +	 * -compatability reasons) a list. +	 */ + +	int elemLen; +	Tcl_Obj **elemPtrs, *copyListObj; + +	result = TclListObjGetElements(interp, arrayElemObj, +		&elemLen, &elemPtrs); +	if (result != TCL_OK) { +	    return result; +	} +	if (elemLen & 1) { +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "list must have an even number of elements", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ARGUMENT", "FORMAT", NULL); +	    return TCL_ERROR; +	} +	if (elemLen == 0) { +	    goto ensureArray; +	} + +	/* +	 * We needn't worry about traces invalidating arrayPtr: should that be +	 * the case, TclPtrSetVar will return NULL so that we break out of the +	 * loop and return an error. +	 */ + +	copyListObj = TclListObjCopy(NULL, arrayElemObj); +	for (i=0 ; i<elemLen ; i+=2) { +	    Var *elemVarPtr = TclLookupArrayElement(interp, arrayNameObj, +		    elemPtrs[i], TCL_LEAVE_ERR_MSG, "set", 1, 1, varPtr, -1); + +	    if ((elemVarPtr == NULL) || +		    (TclPtrSetVar(interp, elemVarPtr, varPtr, arrayNameObj, +		    elemPtrs[i],elemPtrs[i+1],TCL_LEAVE_ERR_MSG,-1) == NULL)){ +		result = TCL_ERROR; +		break; +	    }  	} +	Tcl_DecrRefCount(copyListObj); +	return result;      } -#endif      /* -     * Store the variable's new value and run any write traces. +     * The list is empty make sure we have an array, or create one if +     * necessary.       */ -     -    return TclSetElementOfIndexedArray(interp, localIndex, elemPtr, -	    varValuePtr, TCL_LEAVE_ERR_MSG); + +  ensureArray: +    if (varPtr != NULL) { +	if (TclIsVarArray(varPtr)) { +	    /* +	     * Already an array, done. +	     */ + +	    return TCL_OK; +	} +	if (TclIsVarArrayElement(varPtr) || !TclIsVarUndefined(varPtr)) { +	    /* +	     * Either an array element, or a scalar: lose! +	     */ + +	    TclObjVarErrMsg(interp, arrayNameObj, NULL, "array set", +		    needArray, -1); +	    Tcl_SetErrorCode(interp, "TCL", "WRITE", "ARRAY", NULL); +	    return TCL_ERROR; +	} +    } +    TclSetVarArray(varPtr); +    varPtr->value.tablePtr = ckalloc(sizeof(TclVarHashTable)); +    TclInitVarHashTable(varPtr->value.tablePtr, TclGetVarNsPtr(varPtr)); +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_UnsetVar -- + * ArrayStartSearchCmd --   * - *	Delete a variable, so that it may not be accessed anymore. + *	This object-based function is invoked to process the "array + *	startsearch" Tcl command. See the user documentation for details on + *	what it does.   *   * Results: - *	Returns TCL_OK if the variable was successfully deleted, TCL_ERROR - *	if the variable can't be unset.  In the event of an error, - *	if the TCL_LEAVE_ERR_MSG flag is set then an error message - *	is left in the interp's result. + *	A standard Tcl result object.   *   * Side effects: - *	If varName is defined as a local or global variable in interp, - *	it is deleted. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -int -Tcl_UnsetVar(interp, varName, flags) -    Tcl_Interp *interp;		/* Command interpreter in which varName is -				 * to be looked up. */ -    char *varName;		/* Name of a variable in interp.  May be -				 * either a scalar name or an array name -				 * or an element in an array. */ -    int flags;			/* OR-ed combination of any of -				 * TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY or -				 * TCL_LEAVE_ERR_MSG. */ +	/* ARGSUSED */ +static int +ArrayStartSearchCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    return Tcl_UnsetVar2(interp, varName, (char *) NULL, flags); +    Interp *iPtr = (Interp *) interp; +    Var *varPtr, *arrayPtr; +    Tcl_HashEntry *hPtr; +    Tcl_Obj *varNameObj; +    int isNew; +    ArraySearch *searchPtr; +    const char *varName; + +    if (objc != 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName"); +	return TCL_ERROR; +    } +    varNameObj = objv[1]; + +    /* +     * Locate the array variable. +     */ + +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); +    varName = TclGetString(varNameObj); + +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) { +	    return TCL_ERROR; +	} +    } + +    /* +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces. +     */ + +    if ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)) { +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"\"%s\" isn't an array", varName)); +	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ARRAY", varName, NULL); +	return TCL_ERROR; +    } + +    /* +     * Make a new array search with a free name. +     */ + +    searchPtr = ckalloc(sizeof(ArraySearch)); +    hPtr = Tcl_CreateHashEntry(&iPtr->varSearches, varPtr, &isNew); +    if (isNew) { +	searchPtr->id = 1; +	varPtr->flags |= VAR_SEARCH_ACTIVE; +	searchPtr->nextPtr = NULL; +    } else { +	searchPtr->id = ((ArraySearch *) Tcl_GetHashValue(hPtr))->id + 1; +	searchPtr->nextPtr = Tcl_GetHashValue(hPtr); +    } +    searchPtr->varPtr = varPtr; +    searchPtr->nextEntry = VarHashFirstEntry(varPtr->value.tablePtr, +	    &searchPtr->search); +    Tcl_SetHashValue(hPtr, searchPtr); +    Tcl_SetObjResult(interp, +	    Tcl_ObjPrintf("s-%d-%s", searchPtr->id, varName)); +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_UnsetVar2 -- + * ArrayAnyMoreCmd --   * - *	Delete a variable, given a 2-part name. + *	This object-based function is invoked to process the "array anymore" + *	Tcl command. See the user documentation for details on what it does.   *   * Results: - *	Returns TCL_OK if the variable was successfully deleted, TCL_ERROR - *	if the variable can't be unset.  In the event of an error, - *	if the TCL_LEAVE_ERR_MSG flag is set then an error message - *	is left in the interp's result. + *	A standard Tcl result object.   *   * Side effects: - *	If part1 and part2 indicate a local or global variable in interp, - *	it is deleted.  If part1 is an array name and part2 is NULL, then - *	the whole array is deleted. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -int -Tcl_UnsetVar2(interp, part1, part2, flags) -    Tcl_Interp *interp;		/* Command interpreter in which varName is -				 * to be looked up. */ -    char *part1;		/* Name of variable or array. */ -    char *part2;		/* Name of element within array or NULL. */ -    int flags;			/* OR-ed combination of any of -				 * TCL_GLOBAL_ONLY, TCL_NAMESPACE_ONLY, -				 * TCL_LEAVE_ERR_MSG. */ +	/* ARGSUSED */ +static int +ArrayAnyMoreCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    Var dummyVar; -    Var *varPtr, *dummyVarPtr;      Interp *iPtr = (Interp *) interp; -    Var *arrayPtr; -    ActiveVarTrace *activePtr; -    Tcl_Obj *objPtr; -    int result; +    Var *varPtr, *arrayPtr; +    Tcl_Obj *varNameObj, *searchObj; +    int gotValue; +    ArraySearch *searchPtr; -    varPtr = TclLookupVar(interp, part1, part2, flags, "unset", -	    /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); -    if (varPtr == NULL) { +    if (objc != 3) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName searchId");  	return TCL_ERROR;      } -    result = (TclIsVarUndefined(varPtr)? TCL_ERROR : TCL_OK); - -    if ((arrayPtr != NULL) && (arrayPtr->searchPtr != NULL)) { -	DeleteSearches(arrayPtr); -    } +    varNameObj = objv[1]; +    searchObj = objv[2];      /* -     * The code below is tricky, because of the possibility that -     * a trace procedure might try to access a variable being -     * deleted. To handle this situation gracefully, do things -     * in three steps: -     * 1. Copy the contents of the variable to a dummy variable -     *    structure, and mark the original Var structure as undefined. -     * 2. Invoke traces and clean up the variable, using the dummy copy. -     * 3. If at the end of this the original variable is still -     *    undefined and has no outstanding references, then delete -     *	  it (but it could have gotten recreated by a trace). +     * Locate the array variable.       */ -    dummyVar = *varPtr; -    TclSetVarUndefined(varPtr); -    TclSetVarScalar(varPtr); -    varPtr->value.objPtr = NULL; /* dummyVar points to any value object */ -    varPtr->tracePtr = NULL; -    varPtr->searchPtr = NULL; +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr);      /* -     * Call trace procedures for the variable being deleted. Then delete -     * its traces. Be sure to abort any other traces for the variable -     * that are still pending. Special tricks: -     * 1. We need to increment varPtr's refCount around this: CallTraces -     *    will use dummyVar so it won't increment varPtr's refCount itself. -     * 2. Turn off the VAR_TRACE_ACTIVE flag in dummyVar: we want to -     *    call unset traces even if other traces are pending. +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc.       */ -    if ((dummyVar.tracePtr != NULL) -	    || ((arrayPtr != NULL) && (arrayPtr->tracePtr != NULL))) { -	char *msg; -	int resultType; - -	varPtr->refCount++; -	dummyVar.flags &= ~VAR_TRACE_ACTIVE; -	msg = CallTraces(iPtr, arrayPtr, &dummyVar, part1, part2, -		(flags & (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY)) | TCL_TRACE_UNSETS, -		&resultType); -	if (msg != NULL) { -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *) msg); -	    } -	} -	while (dummyVar.tracePtr != NULL) { -	    VarTrace *tracePtr = dummyVar.tracePtr; -	    dummyVar.tracePtr = tracePtr->nextPtr; -	    Tcl_EventuallyFree((ClientData) tracePtr, TCL_DYNAMIC); -	} -	for (activePtr = iPtr->activeTracePtr;  activePtr != NULL; -	     activePtr = activePtr->nextPtr) { -	    if (activePtr->varPtr == varPtr) { -		activePtr->nextTracePtr = NULL; -	    } +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) { +	    return TCL_ERROR;  	} -	varPtr->refCount--;      }      /* -     * If the variable is an array, delete all of its elements. This must be -     * done after calling the traces on the array, above (that's the way -     * traces are defined). If it is a scalar, "discard" its object -     * (decrement the ref count of its object, if any). +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces.       */ -    dummyVarPtr = &dummyVar; -    if (TclIsVarArray(dummyVarPtr) && !TclIsVarUndefined(dummyVarPtr)) { -	/* -	 * Deleting the elements of the array may cause traces to be fired -	 * on those elements.  Before deleting them, bump the reference count -	 * of the array, so that if those trace procs make a global or upvar -	 * link to the array, the array is not deleted when the call stack -	 * gets popped (we will delete the array ourselves later in this -	 * function). -	 * -	 * Bumping the count can lead to the odd situation that elements of the -	 * array are being deleted when the array still exists, but since the -	 * array is about to be removed anyway, that shouldn't really matter. -	 */ -	varPtr->refCount++; -	DeleteArray(iPtr, part1, dummyVarPtr, -		(flags & (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY)) | TCL_TRACE_UNSETS); -	/* Decr ref count */ -	varPtr->refCount--; -    } -    if (TclIsVarScalar(dummyVarPtr) -	    && (dummyVarPtr->value.objPtr != NULL)) { -	objPtr = dummyVarPtr->value.objPtr; -	TclDecrRefCount(objPtr); -	dummyVarPtr->value.objPtr = NULL; +    if ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)) { +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"\"%s\" isn't an array", TclGetString(varNameObj))); +	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ARRAY", +		TclGetString(varNameObj), NULL); +	return TCL_ERROR;      }      /* -     * If the variable was a namespace variable, decrement its reference count. +     * Get the search.       */ -     -    if (varPtr->flags & VAR_NAMESPACE_VAR) { -	varPtr->flags &= ~VAR_NAMESPACE_VAR; -	varPtr->refCount--; -    } -    /* -     * It's an error to unset an undefined variable. -     */ -	 -    if (result != TCL_OK) { -	if (flags & TCL_LEAVE_ERR_MSG) { -	    VarErrMsg(interp, part1, part2, "unset",  -		    ((arrayPtr == NULL) ? noSuchVar : noSuchElement)); -	} +    searchPtr = ParseSearchId(interp, varPtr, varNameObj, searchObj); +    if (searchPtr == NULL) { +	return TCL_ERROR;      }      /* -     * Finally, if the variable is truly not in use then free up its Var -     * structure and remove it from its hash table, if any. The ref count of -     * its value object, if any, was decremented above. +     * Scan forward to find if there are any further elements in the array +     * that are defined.       */ -    CleanupVar(varPtr, arrayPtr); -    return result; +    while (1) { +	if (searchPtr->nextEntry != NULL) { +	    varPtr = VarHashGetValue(searchPtr->nextEntry); +	    if (!TclIsVarUndefined(varPtr)) { +		gotValue = 1; +		break; +	    } +	} +	searchPtr->nextEntry = Tcl_NextHashEntry(&searchPtr->search); +	if (searchPtr->nextEntry == NULL) { +	    gotValue = 0; +	    break; +	} +    } +    Tcl_SetObjResult(interp, iPtr->execEnvPtr->constants[gotValue]); +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_TraceVar -- + * ArrayNextElementCmd --   * - *	Arrange for reads and/or writes to a variable to cause a - *	procedure to be invoked, which can monitor the operations - *	and/or change their actions. + *	This object-based function is invoked to process the "array + *	nextelement" Tcl command. See the user documentation for details on + *	what it does.   *   * Results: - *	A standard Tcl return value. + *	A standard Tcl result object.   *   * Side effects: - *	A trace is set up on the variable given by varName, such that - *	future references to the variable will be intermediated by - *	proc.  See the manual entry for complete details on the calling - *	sequence for proc. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -int -Tcl_TraceVar(interp, varName, flags, proc, clientData) -    Tcl_Interp *interp;		/* Interpreter in which variable is -				 * to be traced. */ -    char *varName;		/* Name of variable;  may end with "(index)" -				 * to signify an array reference. */ -    int flags;			/* OR-ed collection of bits, including any -				 * of TCL_TRACE_READS, TCL_TRACE_WRITES, -				 * TCL_TRACE_UNSETS, TCL_GLOBAL_ONLY, and -				 * TCL_NAMESPACE_ONLY. */ -    Tcl_VarTraceProc *proc;	/* Procedure to call when specified ops are -				 * invoked upon varName. */ -    ClientData clientData;	/* Arbitrary argument to pass to proc. */ +	/* ARGSUSED */ +static int +ArrayNextElementCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    return Tcl_TraceVar2(interp, varName, (char *) NULL,  -	    flags, proc, clientData); +    Interp *iPtr = (Interp *) interp; +    Var *varPtr, *arrayPtr; +    Tcl_Obj *varNameObj, *searchObj; +    ArraySearch *searchPtr; + +    if (objc != 3) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName searchId"); +	return TCL_ERROR; +    } +    varNameObj = objv[1]; +    searchObj = objv[2]; + +    /* +     * Locate the array variable. +     */ + +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); + +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) { +	    return TCL_ERROR; +	} +    } + +    /* +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces. +     */ + +    if ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)) { +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"\"%s\" isn't an array", TclGetString(varNameObj))); +	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ARRAY", +		TclGetString(varNameObj), NULL); +	return TCL_ERROR; +    } + +    /* +     * Get the search. +     */ + +    searchPtr = ParseSearchId(interp, varPtr, varNameObj, searchObj); +    if (searchPtr == NULL) { +	return TCL_ERROR; +    } + +    /* +     * Get the next element from the search, or the empty string on +     * exhaustion. Note that the [array anymore] command may well have already +     * pulled a value from the hash enumeration, so we have to check the cache +     * there first. +     */ + +    while (1) { +	Tcl_HashEntry *hPtr = searchPtr->nextEntry; + +	if (hPtr == NULL) { +	    hPtr = Tcl_NextHashEntry(&searchPtr->search); +	    if (hPtr == NULL) { +		return TCL_OK; +	    } +	} else { +	    searchPtr->nextEntry = NULL; +	} +	varPtr = VarHashGetValue(hPtr); +	if (!TclIsVarUndefined(varPtr)) { +	    Tcl_SetObjResult(interp, VarHashGetKey(varPtr)); +	    return TCL_OK; +	} +    }  }  /*   *----------------------------------------------------------------------   * - * Tcl_TraceVar2 -- + * ArrayDoneSearchCmd --   * - *	Arrange for reads and/or writes to a variable to cause a - *	procedure to be invoked, which can monitor the operations - *	and/or change their actions. + *	This object-based function is invoked to process the "array + *	donesearch" Tcl command. See the user documentation for details on + *	what it does.   *   * Results: - *	A standard Tcl return value. + *	A standard Tcl result object.   *   * Side effects: - *	A trace is set up on the variable given by part1 and part2, such - *	that future references to the variable will be intermediated by - *	proc.  See the manual entry for complete details on the calling - *	sequence for proc. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -int -Tcl_TraceVar2(interp, part1, part2, flags, proc, clientData) -    Tcl_Interp *interp;		/* Interpreter in which variable is -				 * to be traced. */ -    char *part1;		/* Name of scalar variable or array. */ -    char *part2;		/* Name of element within array;  NULL means -				 * trace applies to scalar variable or array -				 * as-a-whole. */ -    int flags;			/* OR-ed collection of bits, including any -				 * of TCL_TRACE_READS, TCL_TRACE_WRITES, -				 * TCL_TRACE_UNSETS, TCL_GLOBAL_ONLY, -				 * and TCL_NAMESPACE_ONLY. */ -    Tcl_VarTraceProc *proc;	/* Procedure to call when specified ops are -				 * invoked upon varName. */ -    ClientData clientData;	/* Arbitrary argument to pass to proc. */ +	/* ARGSUSED */ +static int +ArrayDoneSearchCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { +    Interp *iPtr = (Interp *) interp;      Var *varPtr, *arrayPtr; -    register VarTrace *tracePtr; -    int flagMask; -     -    /*  -     * We strip 'flags' down to just the parts which are relevant to -     * TclLookupVar, to avoid conflicts between trace flags and -     * internal namespace flags such as 'FIND_ONLY_NS'.  This can -     * now occur since we have trace flags with values 0x1000 and higher. +    Tcl_HashEntry *hPtr; +    Tcl_Obj *varNameObj, *searchObj; +    ArraySearch *searchPtr, *prevPtr; + +    if (objc != 3) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName searchId"); +	return TCL_ERROR; +    } +    varNameObj = objv[1]; +    searchObj = objv[2]; + +    /* +     * Locate the array variable.       */ -    flagMask = TCL_GLOBAL_ONLY | TCL_NAMESPACE_ONLY; -    varPtr = TclLookupVar(interp, part1, part2, -	    (flags & flagMask) | TCL_LEAVE_ERR_MSG, -	    "trace", /*createPart1*/ 1, /*createPart2*/ 1, &arrayPtr); -    if (varPtr == NULL) { + +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); + +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) { +	    return TCL_ERROR; +	} +    } + +    /* +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces. +     */ + +    if ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)) { +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"\"%s\" isn't an array", TclGetString(varNameObj))); +	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ARRAY", +		TclGetString(varNameObj), NULL);  	return TCL_ERROR;      }      /* -     * Check for a nonsense flag combination.  Note that this is a -     * panic() because there should be no code path that ever sets -     * both flags. +     * Get the search.       */ -    if ((flags&TCL_TRACE_RESULT_DYNAMIC) && (flags&TCL_TRACE_RESULT_OBJECT)) { -	panic("bad result flag combination"); + +    searchPtr = ParseSearchId(interp, varPtr, varNameObj, searchObj); +    if (searchPtr == NULL) { +	return TCL_ERROR;      }      /* -     * Set up trace information. +     * Unhook the search from the list of searches associated with the +     * variable.       */ -    flagMask = TCL_TRACE_READS | TCL_TRACE_WRITES | TCL_TRACE_UNSETS |  -	TCL_TRACE_ARRAY | TCL_TRACE_RESULT_DYNAMIC | TCL_TRACE_RESULT_OBJECT; -#ifndef TCL_REMOVE_OBSOLETE_TRACES -    flagMask |= TCL_TRACE_OLD_STYLE; -#endif -    tracePtr = (VarTrace *) ckalloc(sizeof(VarTrace)); -    tracePtr->traceProc		= proc; -    tracePtr->clientData	= clientData; -    tracePtr->flags		= flags & flagMask; -    tracePtr->nextPtr		= varPtr->tracePtr; -    varPtr->tracePtr		= tracePtr; +    hPtr = Tcl_FindHashEntry(&iPtr->varSearches, varPtr); +    if (searchPtr == Tcl_GetHashValue(hPtr)) { +	if (searchPtr->nextPtr) { +	    Tcl_SetHashValue(hPtr, searchPtr->nextPtr); +	} else { +	    varPtr->flags &= ~VAR_SEARCH_ACTIVE; +	    Tcl_DeleteHashEntry(hPtr); +	} +    } else { +	for (prevPtr=Tcl_GetHashValue(hPtr) ;; prevPtr=prevPtr->nextPtr) { +	    if (prevPtr->nextPtr == searchPtr) { +		prevPtr->nextPtr = searchPtr->nextPtr; +		break; +	    } +	} +    } +    ckfree(searchPtr);      return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_UntraceVar -- + * ArrayExistsCmd --   * - *	Remove a previously-created trace for a variable. + *	This object-based function is invoked to process the "array exists" + *	Tcl command. See the user documentation for details on what it does.   *   * Results: - *	None. + *	A standard Tcl result object.   *   * Side effects: - *	If there exists a trace for the variable given by varName - *	with the given flags, proc, and clientData, then that trace - *	is removed. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -void -Tcl_UntraceVar(interp, varName, flags, proc, clientData) -    Tcl_Interp *interp;		/* Interpreter containing variable. */ -    char *varName;		/* Name of variable; may end with "(index)" -				 * to signify an array reference. */ -    int flags;			/* OR-ed collection of bits describing -				 * current trace, including any of -				 * TCL_TRACE_READS, TCL_TRACE_WRITES, -				 * TCL_TRACE_UNSETS, TCL_GLOBAL_ONLY -				 * and TCL_NAMESPACE_ONLY. */ -    Tcl_VarTraceProc *proc;	/* Procedure assocated with trace. */ -    ClientData clientData;	/* Arbitrary argument to pass to proc. */ +	/* ARGSUSED */ +static int +ArrayExistsCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    Tcl_UntraceVar2(interp, varName, (char *) NULL, flags, proc, clientData); +    Interp *iPtr = (Interp *) interp; +    Var *varPtr, *arrayPtr; +    Tcl_Obj *arrayNameObj; +    int notArray; + +    if (objc != 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName"); +	return TCL_ERROR; +    } +    arrayNameObj = objv[1]; + +    /* +     * Locate the array variable. +     */ + +    varPtr = TclObjLookupVarEx(interp, arrayNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); + +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, arrayNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) { +	    return TCL_ERROR; +	} +    } + +    /* +     * Check whether we've actually got an array variable. +     */ + +    notArray = ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)); +    Tcl_SetObjResult(interp, iPtr->execEnvPtr->constants[!notArray]); +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_UntraceVar2 -- + * ArrayGetCmd --   * - *	Remove a previously-created trace for a variable. + *	This object-based function is invoked to process the "array get" Tcl + *	command. See the user documentation for details on what it does.   *   * Results: - *	None. + *	A standard Tcl result object.   *   * Side effects: - *	If there exists a trace for the variable given by part1 - *	and part2 with the given flags, proc, and clientData, then - *	that trace is removed. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -void -Tcl_UntraceVar2(interp, part1, part2, flags, proc, clientData) -    Tcl_Interp *interp;		/* Interpreter containing variable. */ -    char *part1;		/* Name of variable or array. */ -    char *part2;		/* Name of element within array;  NULL means -				 * trace applies to scalar variable or array -				 * as-a-whole. */ -    int flags;			/* OR-ed collection of bits describing -				 * current trace, including any of -				 * TCL_TRACE_READS, TCL_TRACE_WRITES, -				 * TCL_TRACE_UNSETS, TCL_GLOBAL_ONLY, -				 * and TCL_NAMESPACE_ONLY. */ -    Tcl_VarTraceProc *proc;	/* Procedure assocated with trace. */ -    ClientData clientData;	/* Arbitrary argument to pass to proc. */ +	/* ARGSUSED */ +static int +ArrayGetCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    register VarTrace *tracePtr; -    VarTrace *prevPtr; -    Var *varPtr, *arrayPtr;      Interp *iPtr = (Interp *) interp; -    ActiveVarTrace *activePtr; -    int flagMask; -     +    Var *varPtr, *arrayPtr, *varPtr2; +    Tcl_Obj *varNameObj, *nameObj, *valueObj, *nameLstObj, *tmpResObj; +    Tcl_Obj **nameObjPtr, *patternObj; +    Tcl_HashSearch search; +    const char *pattern; +    int i, count, result; + +    switch (objc) { +    case 2: +	varNameObj = objv[1]; +	patternObj = NULL; +	break; +    case 3: +	varNameObj = objv[1]; +	patternObj = objv[2]; +	break; +    default: +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName ?pattern?"); +	return TCL_ERROR; +    } +      /* -     * Set up a mask to mask out the parts of the flags that we are not -     * interested in now. +     * Locate the array variable.       */ -    flagMask = TCL_GLOBAL_ONLY | TCL_NAMESPACE_ONLY; -    varPtr = TclLookupVar(interp, part1, part2, flags & flagMask, -	    /*msg*/ (char *) NULL, -	    /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); -    if (varPtr == NULL) { -	return; -    } +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr);      /* -     * Set up a mask to mask out the parts of the flags that we are not -     * interested in now. +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc.       */ -    flagMask = TCL_TRACE_READS | TCL_TRACE_WRITES | TCL_TRACE_UNSETS | -	TCL_TRACE_ARRAY | TCL_TRACE_RESULT_DYNAMIC | TCL_TRACE_RESULT_OBJECT;  -#ifndef TCL_REMOVE_OBSOLETE_TRACES -    flagMask |= TCL_TRACE_OLD_STYLE; -#endif -    flags &= flagMask; -    for (tracePtr = varPtr->tracePtr, prevPtr = NULL;  ; -	 prevPtr = tracePtr, tracePtr = tracePtr->nextPtr) { -	if (tracePtr == NULL) { -	    return; -	} -	if ((tracePtr->traceProc == proc) && (tracePtr->flags == flags) -		&& (tracePtr->clientData == clientData)) { -	    break; + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) { +	    return TCL_ERROR;  	}      }      /* -     * The code below makes it possible to delete traces while traces -     * are active: it makes sure that the deleted trace won't be -     * processed by CallTraces. +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces. If not an array, it's an empty result. +     */ + +    if ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)) { +	return TCL_OK; +    } + +    pattern = (patternObj ? TclGetString(patternObj) : NULL); + +    /* +     * Store the array names in a new object.       */ -    for (activePtr = iPtr->activeTracePtr;  activePtr != NULL; -	 activePtr = activePtr->nextPtr) { -	if (activePtr->nextTracePtr == tracePtr) { -	    activePtr->nextTracePtr = tracePtr->nextPtr; +    TclNewObj(nameLstObj); +    Tcl_IncrRefCount(nameLstObj); +    if ((patternObj != NULL) && TclMatchIsTrivial(pattern)) { +	varPtr2 = VarHashFindVar(varPtr->value.tablePtr, patternObj); +	if (varPtr2 == NULL) { +	    goto searchDone;  	} +	if (TclIsVarUndefined(varPtr2)) { +	    goto searchDone; +	} +	result = Tcl_ListObjAppendElement(interp, nameLstObj, +		VarHashGetKey(varPtr2)); +	if (result != TCL_OK) { +	    TclDecrRefCount(nameLstObj); +	    return result; +	} +	goto searchDone;      } -    if (prevPtr == NULL) { -	varPtr->tracePtr = tracePtr->nextPtr; -    } else { -	prevPtr->nextPtr = tracePtr->nextPtr; + +    for (varPtr2 = VarHashFirstVar(varPtr->value.tablePtr, &search); +	    varPtr2; varPtr2 = VarHashNextVar(&search)) { +	if (TclIsVarUndefined(varPtr2)) { +	    continue; +	} +	nameObj = VarHashGetKey(varPtr2); +	if (patternObj && !Tcl_StringMatch(TclGetString(nameObj), pattern)) { +	    continue;		/* Element name doesn't match pattern. */ +	} + +	result = Tcl_ListObjAppendElement(interp, nameLstObj, nameObj); +	if (result != TCL_OK) { +	    TclDecrRefCount(nameLstObj); +	    return result; +	}      } -    Tcl_EventuallyFree((ClientData) tracePtr, TCL_DYNAMIC);      /* -     * If this is the last trace on the variable, and the variable is -     * unset and unused, then free up the variable. +     * Make sure the Var structure of the array is not removed by a trace +     * while we're working.       */ -    if (TclIsVarUndefined(varPtr)) { -	CleanupVar(varPtr, (Var *) NULL); +  searchDone: +    if (TclIsVarInHash(varPtr)) { +	VarHashRefCount(varPtr)++; +    } + +    /* +     * Get the array values corresponding to each element name. +     */ + +    TclNewObj(tmpResObj); +    result = Tcl_ListObjGetElements(interp, nameLstObj, &count, &nameObjPtr); +    if (result != TCL_OK) { +	goto errorInArrayGet; +    } + +    for (i=0 ; i<count ; i++) { +	nameObj = *nameObjPtr++; +	valueObj = Tcl_ObjGetVar2(interp, varNameObj, nameObj, +		TCL_LEAVE_ERR_MSG); +	if (valueObj == NULL) { +	    /* +	     * Some trace played a trick on us; we need to diagnose to adapt +	     * our behaviour: was the array element unset, or did the +	     * modification modify the complete array? +	     */ + +	    if (TclIsVarArray(varPtr)) { +		/* +		 * The array itself looks OK, the variable was undefined: +		 * forget it. +		 */ + +		continue; +	    } +	    result = TCL_ERROR; +	    goto errorInArrayGet; +	} +	result = Tcl_DictObjPut(interp, tmpResObj, nameObj, valueObj); +	if (result != TCL_OK) { +	    goto errorInArrayGet; +	}      } +    if (TclIsVarInHash(varPtr)) { +	VarHashRefCount(varPtr)--; +    } +    Tcl_SetObjResult(interp, tmpResObj); +    TclDecrRefCount(nameLstObj); +    return TCL_OK; + +  errorInArrayGet: +    if (TclIsVarInHash(varPtr)) { +	VarHashRefCount(varPtr)--; +    } +    TclDecrRefCount(nameLstObj); +    TclDecrRefCount(tmpResObj);	/* Free unneeded temp result. */ +    return result;  }  /*   *----------------------------------------------------------------------   * - * Tcl_VarTraceInfo -- + * ArrayNamesCmd --   * - *	Return the clientData value associated with a trace on a - *	variable.  This procedure can also be used to step through - *	all of the traces on a particular variable that have the - *	same trace procedure. + *	This object-based function is invoked to process the "array names" Tcl + *	command. See the user documentation for details on what it does.   *   * Results: - *	The return value is the clientData value associated with - *	a trace on the given variable.  Information will only be - *	returned for a trace with proc as trace procedure.  If - *	the clientData argument is NULL then the first such trace is - *	returned;  otherwise, the next relevant one after the one - *	given by clientData will be returned.  If the variable - *	doesn't exist, or if there are no (more) traces for it, - *	then NULL is returned. + *	A standard Tcl result object.   *   * Side effects: - *	None. + *	See the user documentation.   *   *----------------------------------------------------------------------   */ -ClientData -Tcl_VarTraceInfo(interp, varName, flags, proc, prevClientData) -    Tcl_Interp *interp;		/* Interpreter containing variable. */ -    char *varName;		/* Name of variable;  may end with "(index)" -				 * to signify an array reference. */ -    int flags;			/* OR-ed combo or TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY (can be 0). */ -    Tcl_VarTraceProc *proc;	/* Procedure assocated with trace. */ -    ClientData prevClientData;	/* If non-NULL, gives last value returned -				 * by this procedure, so this call will -				 * return the next trace after that one. -				 * If NULL, this call will return the -				 * first trace. */ +	/* ARGSUSED */ +static int +ArrayNamesCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    return Tcl_VarTraceInfo2(interp, varName, (char *) NULL, -	    flags, proc, prevClientData); +    static const char *const options[] = { +	"-exact", "-glob", "-regexp", NULL +    }; +    enum options { OPT_EXACT, OPT_GLOB, OPT_REGEXP }; +    Interp *iPtr = (Interp *) interp; +    Var *varPtr, *arrayPtr, *varPtr2; +    Tcl_Obj *varNameObj, *nameObj, *resultObj, *patternObj; +    Tcl_HashSearch search; +    const char *pattern = NULL; +    int mode = OPT_GLOB; + +    if ((objc < 2) || (objc > 4)) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName ?mode? ?pattern?"); +	return TCL_ERROR; +    } +    varNameObj = objv[1]; +    patternObj = (objc > 2 ? objv[objc-1] : NULL); + +    /* +     * Locate the array variable. +     */ + +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); + +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) { +	    return TCL_ERROR; +	} +    } + +    /* +     * Finish parsing the arguments. +     */ + +    if ((objc == 4) && Tcl_GetIndexFromObj(interp, objv[2], options, "option", +	    0, &mode) != TCL_OK) { +	return TCL_ERROR; +    } + +    /* +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces. If not an array, the result is empty. +     */ + +    if ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)) { +	return TCL_OK; +    } + +    /* +     * Check for the trivial cases where we can use a direct lookup. +     */ + +    TclNewObj(resultObj); +    if (patternObj) { +	pattern = TclGetString(patternObj); +    } +    if ((mode==OPT_GLOB && patternObj && TclMatchIsTrivial(pattern)) +	    || (mode==OPT_EXACT)) { +	varPtr2 = VarHashFindVar(varPtr->value.tablePtr, patternObj); +	if ((varPtr2 != NULL) && !TclIsVarUndefined(varPtr2)) { +	    /* +	     * This can't fail; lappending to an empty object always works. +	     */ + +	    Tcl_ListObjAppendElement(NULL, resultObj, VarHashGetKey(varPtr2)); +	} +	Tcl_SetObjResult(interp, resultObj); +	return TCL_OK; +    } + +    /* +     * Must scan the array to select the elements. +     */ + +    for (varPtr2=VarHashFirstVar(varPtr->value.tablePtr, &search); +	    varPtr2!=NULL ; varPtr2=VarHashNextVar(&search)) { +	if (TclIsVarUndefined(varPtr2)) { +	    continue; +	} +	nameObj = VarHashGetKey(varPtr2); +	if (patternObj) { +	    const char *name = TclGetString(nameObj); +	    int matched = 0; + +	    switch ((enum options) mode) { +	    case OPT_EXACT: +		Tcl_Panic("exact matching shouldn't get here"); +	    case OPT_GLOB: +		matched = Tcl_StringMatch(name, pattern); +		break; +	    case OPT_REGEXP: +		matched = Tcl_RegExpMatchObj(interp, nameObj, patternObj); +		if (matched < 0) { +		    TclDecrRefCount(resultObj); +		    return TCL_ERROR; +		} +		break; +	    } +	    if (matched == 0) { +		continue; +	    } +	} + +	Tcl_ListObjAppendElement(NULL, resultObj, nameObj); +    } +    Tcl_SetObjResult(interp, resultObj); +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_VarTraceInfo2 -- + * TclFindArrayPtrElements --   * - *	Same as Tcl_VarTraceInfo, except takes name in two pieces - *	instead of one. + *	Fill out a hash table (which *must* use Tcl_Obj* keys) with an entry + *	for each existing element of the given array. The provided hash table + *	is assumed to be initially empty.   * - * Results: - *	Same as Tcl_VarTraceInfo. + * Result: + *	none   *   * Side effects: - *	None. + *	The keys of the array gain an extra reference. The supplied hash table + *	has elements added to it.   *   *----------------------------------------------------------------------   */ -ClientData -Tcl_VarTraceInfo2(interp, part1, part2, flags, proc, prevClientData) -    Tcl_Interp *interp;		/* Interpreter containing variable. */ -    char *part1;		/* Name of variable or array. */ -    char *part2;		/* Name of element within array;  NULL means -				 * trace applies to scalar variable or array -				 * as-a-whole. */ -    int flags;			/* OR-ed combination of TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY. */ -    Tcl_VarTraceProc *proc;	/* Procedure assocated with trace. */ -    ClientData prevClientData;	/* If non-NULL, gives last value returned -				 * by this procedure, so this call will -				 * return the next trace after that one. -				 * If NULL, this call will return the -				 * first trace. */ +void +TclFindArrayPtrElements( +    Var *arrayPtr, +    Tcl_HashTable *tablePtr)  { -    register VarTrace *tracePtr; -    Var *varPtr, *arrayPtr; +    Var *varPtr; +    Tcl_HashSearch search; -    varPtr = TclLookupVar(interp, part1, part2, -	    flags & (TCL_GLOBAL_ONLY|TCL_NAMESPACE_ONLY), -	    /*msg*/ (char *) NULL, -	    /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); -    if (varPtr == NULL) { -	return NULL; +    if ((arrayPtr == NULL) || !TclIsVarArray(arrayPtr) +	    || TclIsVarUndefined(arrayPtr)) { +	return;      } -    /* -     * Find the relevant trace, if any, and return its clientData. -     */ +    for (varPtr=VarHashFirstVar(arrayPtr->value.tablePtr, &search); +	    varPtr!=NULL ; varPtr=VarHashNextVar(&search)) { +	Tcl_HashEntry *hPtr; +	Tcl_Obj *nameObj; +	int dummy; -    tracePtr = varPtr->tracePtr; -    if (prevClientData != NULL) { -	for ( ;  tracePtr != NULL;  tracePtr = tracePtr->nextPtr) { -	    if ((tracePtr->clientData == prevClientData) -		    && (tracePtr->traceProc == proc)) { -		tracePtr = tracePtr->nextPtr; -		break; -	    } -	} -    } -    for ( ;  tracePtr != NULL;  tracePtr = tracePtr->nextPtr) { -	if (tracePtr->traceProc == proc) { -	    return tracePtr->clientData; +	if (TclIsVarUndefined(varPtr)) { +	    continue;  	} +	nameObj = VarHashGetKey(varPtr); +	hPtr = Tcl_CreateHashEntry(tablePtr, (char *) nameObj, &dummy); +	Tcl_SetHashValue(hPtr, nameObj);      } -    return NULL;  }  /*   *----------------------------------------------------------------------   * - * Tcl_UnsetObjCmd -- + * ArraySetCmd --   * - *	This object-based procedure is invoked to process the "unset" Tcl + *	This object-based function is invoked to process the "array set" Tcl   *	command. See the user documentation for details on what it does.   *   * Results: - *	A standard Tcl object result value. + *	A standard Tcl result object.   *   * Side effects:   *	See the user documentation. @@ -2895,296 +3912,218 @@ Tcl_VarTraceInfo2(interp, part1, part2, flags, proc, prevClientData)   */  	/* ARGSUSED */ -int -Tcl_UnsetObjCmd(dummy, interp, objc, objv) -    ClientData dummy;		/* Not used. */ -    Tcl_Interp *interp;		/* Current interpreter. */ -    int objc;			/* Number of arguments. */ -    Tcl_Obj *CONST objv[];	/* Argument objects. */ +static int +ArraySetCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    register int i, flags = TCL_LEAVE_ERR_MSG; -    register char *name; +    Interp *iPtr = (Interp *) interp; +    Var *varPtr, *arrayPtr; -    if (objc < 1) { -	Tcl_WrongNumArgs(interp, 1, objv, -		"?-nocomplain? ?--? ?varName varName ...?"); +    if (objc != 3) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName list");  	return TCL_ERROR; -    } else if (objc == 1) { -	/* -	 * Do nothing if no arguments supplied, so as to match -	 * command documentation. -	 */ -	return TCL_OK;      }      /* -     * Simple, restrictive argument parsing.  The only options are -- -     * and -nocomplain (which must come first and be given exactly to -     * be an option). +     * Locate the array variable.       */ -    i = 1; -    name = TclGetString(objv[i]); -    if (name[0] == '-') { - 	if (strcmp("-nocomplain", name) == 0) { -	    i++; - 	    if (i == objc) { -		return TCL_OK; -	    } - 	    flags = 0; - 	    name = TclGetString(objv[i]); - 	} - 	if (strcmp("--", name) == 0) { - 	    i++; - 	} -    } -    for (; i < objc;  i++) { -	name = TclGetString(objv[i]); -	if ((Tcl_UnsetVar2(interp, name, (char *) NULL, flags) != TCL_OK) -		&& (flags == TCL_LEAVE_ERR_MSG)) { +    varPtr = TclObjLookupVarEx(interp, objv[1], NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); + +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, objv[1], NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) {  	    return TCL_ERROR;  	}      } -    return TCL_OK; + +    return TclArraySet(interp, objv[1], objv[2]);  }  /*   *----------------------------------------------------------------------   * - * Tcl_AppendObjCmd -- + * ArraySizeCmd --   * - *	This object-based procedure is invoked to process the "append"  - *	Tcl command. See the user documentation for details on what it does. + *	This object-based function is invoked to process the "array size" Tcl + *	command. See the user documentation for details on what it does.   *   * Results: - *	A standard Tcl object result value. + *	A standard Tcl result object.   *   * Side effects: - *	A variable's value may be changed. + *	See the user documentation.   *   *----------------------------------------------------------------------   */  	/* ARGSUSED */ -int -Tcl_AppendObjCmd(dummy, interp, objc, objv) -    ClientData dummy;		/* Not used. */ -    Tcl_Interp *interp;		/* Current interpreter. */ -    int objc;			/* Number of arguments. */ -    Tcl_Obj *CONST objv[];	/* Argument objects. */ +static int +ArraySizeCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    register Tcl_Obj *varValuePtr = NULL; -    					/* Initialized to avoid compiler -				         * warning. */ -    int i; +    Interp *iPtr = (Interp *) interp; +    Var *varPtr, *arrayPtr; +    Tcl_Obj *varNameObj; +    Tcl_HashSearch search; +    Var *varPtr2; +    int size = 0; -    if (objc < 2) { -	Tcl_WrongNumArgs(interp, 1, objv, "varName ?value value ...?"); +    if (objc != 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName");  	return TCL_ERROR;      } -    if (objc == 2) { -	varValuePtr = Tcl_ObjGetVar2(interp, objv[1], NULL, TCL_LEAVE_ERR_MSG); -	if (varValuePtr == NULL) { +    varNameObj = objv[1]; + +    /* +     * Locate the array variable. +     */ + +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); + +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ + +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) {  	    return TCL_ERROR;  	} -    } else { -	for (i = 2;  i < objc;  i++) { -	    varValuePtr = Tcl_ObjSetVar2(interp, objv[1], (Tcl_Obj *) NULL, -		    objv[i], (TCL_APPEND_VALUE | TCL_LEAVE_ERR_MSG)); -	    if (varValuePtr == NULL) { -		return TCL_ERROR; +    } + +    /* +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces. We can only iterate over the array if it exists... +     */ + +    if (varPtr && TclIsVarArray(varPtr) && !TclIsVarUndefined(varPtr)) { +	/* +	 * Must iterate in order to get chance to check for present but +	 * "undefined" entries. +	 */ + +	for (varPtr2=VarHashFirstVar(varPtr->value.tablePtr, &search); +		varPtr2!=NULL ; varPtr2=VarHashNextVar(&search)) { +	    if (!TclIsVarUndefined(varPtr2)) { +		size++;  	    }  	}      } -    Tcl_SetObjResult(interp, varValuePtr); + +    Tcl_SetObjResult(interp, Tcl_NewIntObj(size));      return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_LappendObjCmd -- + * ArrayStatsCmd --   * - *	This object-based procedure is invoked to process the "lappend"  - *	Tcl command. See the user documentation for details on what it does. + *	This object-based function is invoked to process the "array + *	statistics" Tcl command. See the user documentation for details on + *	what it does.   *   * Results: - *	A standard Tcl object result value. + *	A standard Tcl result object.   *   * Side effects: - *	A variable's value may be changed. + *	See the user documentation.   *   *----------------------------------------------------------------------   */  	/* ARGSUSED */ -int -Tcl_LappendObjCmd(dummy, interp, objc, objv) -    ClientData dummy;		/* Not used. */ -    Tcl_Interp *interp;		/* Current interpreter. */ -    int objc;			/* Number of arguments. */ -    Tcl_Obj *CONST objv[];	/* Argument objects. */ +static int +ArrayStatsCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    Tcl_Obj *varValuePtr, *newValuePtr; -    register List *listRepPtr; -    register Tcl_Obj **elemPtrs; -    int numElems, numRequired, createdNewObj, createVar, i, j; +    Interp *iPtr = (Interp *) interp; +    Var *varPtr, *arrayPtr; +    Tcl_Obj *varNameObj; +    char *stats; -    if (objc < 2) { -	Tcl_WrongNumArgs(interp, 1, objv, "varName ?value value ...?"); +    if (objc != 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName");  	return TCL_ERROR;      } -    if (objc == 2) { -	newValuePtr = Tcl_ObjGetVar2(interp, objv[1], (Tcl_Obj *) NULL, 0); -	if (newValuePtr == NULL) { -	    /* -	     * The variable doesn't exist yet. Just create it with an empty -	     * initial value. -	     */ -	     -	    varValuePtr = Tcl_NewObj(); -	    newValuePtr = Tcl_ObjSetVar2(interp, objv[1], NULL, varValuePtr, -		    TCL_LEAVE_ERR_MSG); -	    if (newValuePtr == NULL) { -		Tcl_DecrRefCount(varValuePtr); /* free unneeded object */ -		return TCL_ERROR; -	    } -	} -    } else { -	/* -	 * We have arguments to append. We used to call Tcl_SetVar2 to -	 * append each argument one at a time to ensure that traces were run -	 * for each append step. We now append the arguments all at once -	 * because it's faster. Note that a read trace and a write trace for -	 * the variable will now each only be called once. Also, if the -	 * variable's old value is unshared we modify it directly, otherwise -	 * we create a new copy to modify: this is "copy on write". -	 */ - -	createdNewObj = 0; -	createVar = 1; -	/* -	 * Use the TCL_TRACE_READS flag to ensure that if we have an -	 * array with no elements set yet, but with a read trace on it, -	 * we will create the variable and get read traces triggered. -	 */ -	varValuePtr = Tcl_ObjGetVar2(interp, objv[1], NULL, TCL_TRACE_READS); -	if (varValuePtr == NULL) { -	    /* -	     * We couldn't read the old value: either the var doesn't yet -	     * exist or it's an array element.  If it's new, we will try to -	     * create it with Tcl_ObjSetVar2 below. -	     */ -	     -	    char *p, *varName; -	    int nameBytes, i; - -	    varName = Tcl_GetStringFromObj(objv[1], &nameBytes); -	    for (i = 0, p = varName;  i < nameBytes;  i++, p++) { -		if (*p == '(') { -		    p = (varName + nameBytes-1);	 -		    if (*p == ')') { /* last char is ')' => array ref */ -			/* -			 * This case occurs when we tried something like: -			 set x "" -			 lappend x(0) 44 -			 */ -			createVar = 0; -		    } -		    break; -		} -	    } -	    varValuePtr = Tcl_NewObj(); -	    createdNewObj = 1; -	} else if (Tcl_IsShared(varValuePtr)) {	 -	    varValuePtr = Tcl_DuplicateObj(varValuePtr); -	    createdNewObj = 1; -	} - -	/* -	 * Convert the variable's old value to a list object if necessary. -	 */ - -	if (varValuePtr->typePtr != &tclListType) { -	    int result = tclListType.setFromAnyProc(interp, varValuePtr); -	    if (result != TCL_OK) { -		if (createdNewObj) { -		    Tcl_DecrRefCount(varValuePtr); /* free unneeded obj. */ -		} -		return result; -	    } -	} -	listRepPtr = (List *) varValuePtr->internalRep.twoPtrValue.ptr1; -	elemPtrs = listRepPtr->elements; -	numElems = listRepPtr->elemCount; - -	/* -	 * If there is no room in the current array of element pointers, -	 * allocate a new, larger array and copy the pointers to it. -	 */ -	 -	numRequired = numElems + (objc-2); -	if (numRequired > listRepPtr->maxElemCount) { -	    int newMax = (2 * numRequired); -	    Tcl_Obj **newElemPtrs = (Tcl_Obj **) -		ckalloc((unsigned) (newMax * sizeof(Tcl_Obj *))); -	     -	    memcpy((VOID *) newElemPtrs, (VOID *) elemPtrs, -		    (size_t) (numElems * sizeof(Tcl_Obj *))); -	    listRepPtr->maxElemCount = newMax; -	    listRepPtr->elements = newElemPtrs; -	    ckfree((char *) elemPtrs); -	    elemPtrs = newElemPtrs; -	} - -	/* -	 * Insert the new elements at the end of the list. -	 */ +    varNameObj = objv[1]; -	for (i = 2, j = numElems;  i < objc;  i++, j++) { -            elemPtrs[j] = objv[i]; -            Tcl_IncrRefCount(objv[i]); -        } -	listRepPtr->elemCount = numRequired; +    /* +     * Locate the array variable. +     */ -	/* -	 * Invalidate and free any old string representation since it no -	 * longer reflects the list's internal representation. -	 */ +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); -	Tcl_InvalidateStringRep(varValuePtr); +    /* +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc. +     */ -	/* -	 * Now store the list object back into the variable. If there is an -	 * error setting the new value, decrement its ref count if it -	 * was new and we didn't create the variable. -	 */ -	 -	newValuePtr = Tcl_ObjSetVar2(interp, objv[1], NULL, varValuePtr, -		TCL_LEAVE_ERR_MSG); -	if (newValuePtr == NULL) { -	    if (createdNewObj && !createVar) { -		Tcl_DecrRefCount(varValuePtr); /* free unneeded obj */ -	    } +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY) +	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL, +		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) {  	    return TCL_ERROR;  	}      }      /* -     * Set the interpreter's object result to refer to the variable's value -     * object. +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces.       */ -    Tcl_SetObjResult(interp, newValuePtr); +    if ((varPtr == NULL) || !TclIsVarArray(varPtr) +	    || TclIsVarUndefined(varPtr)) { +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"\"%s\" isn't an array", TclGetString(varNameObj))); +	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ARRAY", +		TclGetString(varNameObj), NULL); +	return TCL_ERROR; +    } + +    stats = Tcl_HashStats((Tcl_HashTable *) varPtr->value.tablePtr); +    if (stats == NULL) { +	Tcl_SetObjResult(interp, Tcl_NewStringObj( +		"error reading array statistics", -1)); +	return TCL_ERROR; +    } +    Tcl_SetObjResult(interp, Tcl_NewStringObj(stats, -1)); +    ckfree(stats);      return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * Tcl_ArrayObjCmd -- + * ArrayUnsetCmd --   * - *	This object-based procedure is invoked to process the "array" Tcl + *	This object-based function is invoked to process the "array unset" Tcl   *	command. See the user documentation for details on what it does.   *   * Results: @@ -3197,827 +4136,446 @@ Tcl_LappendObjCmd(dummy, interp, objc, objv)   */  	/* ARGSUSED */ -int -Tcl_ArrayObjCmd(dummy, interp, objc, objv) -    ClientData dummy;		/* Not used. */ -    Tcl_Interp *interp;		/* Current interpreter. */ -    int objc;			/* Number of arguments. */ -    Tcl_Obj *CONST objv[];	/* Argument objects. */ +static int +ArrayUnsetCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[])  { -    /* -     * The list of constants below should match the arrayOptions string array -     * below. -     */ - -    enum {ARRAY_ANYMORE, ARRAY_DONESEARCH,  ARRAY_EXISTS, ARRAY_GET, -	  ARRAY_NAMES, ARRAY_NEXTELEMENT, ARRAY_SET, ARRAY_SIZE, -	  ARRAY_STARTSEARCH, ARRAY_STATISTICS, ARRAY_UNSET};  -    static CONST char *arrayOptions[] = { -	"anymore", "donesearch", "exists", "get", "names", "nextelement", -	"set", "size", "startsearch", "statistics", "unset", (char *) NULL -    }; -      Interp *iPtr = (Interp *) interp; -    Var *varPtr, *arrayPtr; -    Tcl_HashEntry *hPtr; -    Tcl_Obj *resultPtr; -    int notArray; -    char *varName, *msg; -    int index, result; - - -    if (objc < 3) { -	Tcl_WrongNumArgs(interp, 1, objv, "option arrayName ?arg ...?"); +    Var *varPtr, *arrayPtr, *varPtr2, *protectedVarPtr; +    Tcl_Obj *varNameObj, *patternObj, *nameObj; +    Tcl_HashSearch search; +    const char *pattern; +    const int unsetFlags = 0;	/* Should this be TCL_LEAVE_ERR_MSG? */ + +    switch (objc) { +    case 2: +	varNameObj = objv[1]; +	patternObj = NULL; +	break; +    case 3: +	varNameObj = objv[1]; +	patternObj = objv[2]; +	break; +    default: +	Tcl_WrongNumArgs(interp, 1, objv, "arrayName ?pattern?");  	return TCL_ERROR;      } -    if (Tcl_GetIndexFromObj(interp, objv[1], arrayOptions, "option", -	    0, &index) != TCL_OK) { -    	return TCL_ERROR; -    } -      /*       * Locate the array variable       */ -     -    varName = TclGetString(objv[2]); -    varPtr = TclLookupVar(interp, varName, (char *) NULL, /*flags*/ 0, -            /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); + +    varPtr = TclObjLookupVarEx(interp, varNameObj, NULL, /*flags*/ 0, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr);      /* -     * Special array trace used to keep the env array in sync for -     * array names, array get, etc. +     * Special array trace used to keep the env array in sync for array names, +     * array get, etc.       */ -    if (varPtr != NULL && varPtr->tracePtr != NULL +    if (varPtr && (varPtr->flags & VAR_TRACED_ARRAY)  	    && (TclIsVarArray(varPtr) || TclIsVarUndefined(varPtr))) { -	int resultType; - -	msg = CallTraces(iPtr, arrayPtr, varPtr, varName, NULL, +	if (TclObjCallVarTraces(iPtr, arrayPtr, varPtr, varNameObj, NULL,  		(TCL_LEAVE_ERR_MSG|TCL_NAMESPACE_ONLY|TCL_GLOBAL_ONLY| -		TCL_TRACE_ARRAY), &resultType); -	if (msg != NULL) { -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		VarErrMsg(interp, varName, NULL, "trace array", msg); -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		VarErrMsg(interp, varName, NULL, "trace array", -			  Tcl_GetString((Tcl_Obj *) msg)); -		Tcl_DecrRefCount((Tcl_Obj *)msg); -	    } else { -		VarErrMsg(interp, varName, NULL, "trace array", msg); -	    } +		TCL_TRACE_ARRAY), /* leaveErrMsg */ 1, -1) == TCL_ERROR) {  	    return TCL_ERROR;  	}      }      /* -     * Verify that it is indeed an array variable. This test comes after -     * the traces - the variable may actually become an array as an effect  -     * of said traces. +     * Verify that it is indeed an array variable. This test comes after the +     * traces - the variable may actually become an array as an effect of said +     * traces.       */ -    notArray = 0;      if ((varPtr == NULL) || !TclIsVarArray(varPtr)  	    || TclIsVarUndefined(varPtr)) { -	notArray = 1; +	return TCL_OK;      } -    /* -     * We have to wait to get the resultPtr until here because -     * CallTraces can affect the result. -     */ - -    resultPtr = Tcl_GetObjResult(interp); +    if (!patternObj) { +	/* +	 * When no pattern is given, just unset the whole array. +	 */ -    switch (index) { -        case ARRAY_ANYMORE: { -	    ArraySearch *searchPtr; -	     -	    if (objc != 4) { -	        Tcl_WrongNumArgs(interp, 2, objv,  -                        "arrayName searchId"); -		return TCL_ERROR; -	    } -	    if (notArray) { -	        goto error; -	    } -	    searchPtr = ParseSearchId(interp, varPtr, varName, objv[3]); -	    if (searchPtr == NULL) { -	        return TCL_ERROR; -	    } -	    while (1) { -	        Var *varPtr2; +	return TclObjUnsetVar2(interp, varNameObj, NULL, 0); +    } -		if (searchPtr->nextEntry != NULL) { -		    varPtr2 = (Var *) Tcl_GetHashValue(searchPtr->nextEntry); -		    if (!TclIsVarUndefined(varPtr2)) { -		        break; -		    } -		} -		searchPtr->nextEntry = Tcl_NextHashEntry(&searchPtr->search); -		if (searchPtr->nextEntry == NULL) { -		    Tcl_SetIntObj(resultPtr, 0); -		    return TCL_OK; -		} -	    } -	    Tcl_SetIntObj(resultPtr, 1); -	    break; -	} -        case ARRAY_DONESEARCH: { -	    ArraySearch *searchPtr, *prevPtr; +    /* +     * With a trivial pattern, we can just unset. +     */ -	    if (objc != 4) { -	        Tcl_WrongNumArgs(interp, 2, objv,  -                        "arrayName searchId"); -		return TCL_ERROR; -	    } -	    if (notArray) { -	        goto error; -	    } -	    searchPtr = ParseSearchId(interp, varPtr, varName, objv[3]); -	    if (searchPtr == NULL) { -	        return TCL_ERROR; -	    } -	    if (varPtr->searchPtr == searchPtr) { -	        varPtr->searchPtr = searchPtr->nextPtr; -	    } else { -	        for (prevPtr = varPtr->searchPtr;  ; -		     prevPtr = prevPtr->nextPtr) { -		    if (prevPtr->nextPtr == searchPtr) { -		        prevPtr->nextPtr = searchPtr->nextPtr; -			break; -		    } -		} -	    } -	    ckfree((char *) searchPtr); -	    break; -	} -        case ARRAY_EXISTS: { -	    if (objc != 3) { -	        Tcl_WrongNumArgs(interp, 2, objv, "arrayName"); -	        return TCL_ERROR; -	    } -	    Tcl_SetIntObj(resultPtr, !notArray); -	    break; +    pattern = TclGetString(patternObj); +    if (TclMatchIsTrivial(pattern)) { +	varPtr2 = VarHashFindVar(varPtr->value.tablePtr, patternObj); +	if (!varPtr2 || TclIsVarUndefined(varPtr2)) { +	    return TCL_OK;  	} -        case ARRAY_GET: { -	    Tcl_HashSearch search; -	    Var *varPtr2; -	    char *pattern = NULL; -	    char *name; -	    Tcl_Obj *namePtr, *valuePtr, *nameLstPtr, *tmpResPtr, **namePtrPtr; -	    int i, count; -	     -	    if ((objc != 3) && (objc != 4)) { -	        Tcl_WrongNumArgs(interp, 2, objv, "arrayName ?pattern?"); -		return TCL_ERROR; -	    } -	    if (notArray) { -	        return TCL_OK; -	    } -	    if (objc == 4) { -	        pattern = TclGetString(objv[3]); -	    } - -	    /* -	     * Store the array names in a new object. -	     */ - -	    nameLstPtr = Tcl_NewObj(); -	    Tcl_IncrRefCount(nameLstPtr); - -	    for (hPtr = Tcl_FirstHashEntry(varPtr->value.tablePtr, &search); -		 hPtr != NULL;  hPtr = Tcl_NextHashEntry(&search)) { -	        varPtr2 = (Var *) Tcl_GetHashValue(hPtr); -		if (TclIsVarUndefined(varPtr2)) { -		    continue; -		} -		name = Tcl_GetHashKey(varPtr->value.tablePtr, hPtr); -		if ((objc == 4) && !Tcl_StringMatch(name, pattern)) { -		    continue;	/* element name doesn't match pattern */ -		} -		 -		namePtr = Tcl_NewStringObj(name, -1); -		result = Tcl_ListObjAppendElement(interp, nameLstPtr, -		        namePtr); -		if (result != TCL_OK) { -		    Tcl_DecrRefCount(namePtr); /* free unneeded name obj */ -		    Tcl_DecrRefCount(nameLstPtr); -		    return result; -		} -	    } - -	    /* -	     * Make sure the Var structure of the array is not removed by -	     * a trace while we're working. -	     */ - -	    varPtr->refCount++; - -	    /* -	     * Get the array values corresponding to each element name  -	     */ +	return TclPtrUnsetVar(interp, varPtr2, varPtr, varNameObj, patternObj, +		unsetFlags, -1); +    } -	    tmpResPtr = Tcl_NewObj(); -	    result = Tcl_ListObjGetElements(interp, nameLstPtr, -		    &count, &namePtrPtr); -	    if (result != TCL_OK) { -		goto errorInArrayGet; -	    } -	     -	    for (i = 0; i < count; i++) {  -		namePtr = *namePtrPtr++; -		valuePtr = Tcl_ObjGetVar2(interp, objv[2], namePtr, -	                TCL_LEAVE_ERR_MSG); -		if (valuePtr == NULL) { -		    /* -		     * Some trace played a trick on us; we need to diagnose to -		     * adapt our behaviour: was the array element unset, or did -		     * the modification modify the complete array? -		     */ +    /* +     * Non-trivial case (well, deeply tricky really). We peek inside the hash +     * iterator in order to allow us to guarantee that the following element +     * in the array will not be scrubbed until we have dealt with it. This +     * stops the overall iterator from ending up pointing into deallocated +     * memory. [Bug 2939073] +     */ -		    if (TclIsVarArray(varPtr) && !TclIsVarUndefined(varPtr)) { -			/* -			 * The array itself looks OK, the variable was -			 * undefined: forget it. -			 */ -			 -			continue; -		    } else { -			result = TCL_ERROR; -			goto errorInArrayGet; -		    } -		} -		result = Tcl_ListObjAppendElement(interp, tmpResPtr, namePtr); -		if (result != TCL_OK) { -		    goto errorInArrayGet; -		} -		result = Tcl_ListObjAppendElement(interp, tmpResPtr, valuePtr); -		if (result != TCL_OK) { -		    goto errorInArrayGet; -		} -	    } -	    varPtr->refCount--; -	    Tcl_SetObjResult(interp, tmpResPtr); -	    Tcl_DecrRefCount(nameLstPtr); -	    break; +    protectedVarPtr = NULL; +    for (varPtr2=VarHashFirstVar(varPtr->value.tablePtr, &search); +	    varPtr2!=NULL ; varPtr2=VarHashNextVar(&search)) { +	/* +	 * Drop the extra ref immediately. We don't need to free it at this +	 * point though; we'll be unsetting it if necessary soon. +	 */ -	    errorInArrayGet: -	    varPtr->refCount--; -	    Tcl_DecrRefCount(nameLstPtr); -	    Tcl_DecrRefCount(tmpResPtr); /* free unneeded temp result obj */ -	    return result; -	} -        case ARRAY_NAMES: { -	    Tcl_HashSearch search; -	    Var *varPtr2; -	    char *pattern = NULL; -	    char *name; -	    Tcl_Obj *namePtr; -	    int mode, matched = 0; -	    static CONST char *options[] = { -		"-exact", "-glob", "-regexp", (char *) NULL -	    }; -	    enum options { OPT_EXACT, OPT_GLOB, OPT_REGEXP }; - -	    mode = OPT_GLOB; -	     -	    if ((objc < 3) && (objc > 5)) { -  	        Tcl_WrongNumArgs(interp, 2, objv, -			"arrayName ?mode? ?pattern?"); -		return TCL_ERROR; -	    } -	    if (notArray) { -	        return TCL_OK; -	    } -	    if (objc == 4) { -	        pattern = Tcl_GetString(objv[3]); -	    } else if (objc == 5) { -		pattern = Tcl_GetString(objv[4]); -		if (Tcl_GetIndexFromObj(interp, objv[3], options, "option", -			0, &mode) != TCL_OK) { -		    return TCL_ERROR; -		} -	    }       		 -	    for (hPtr = Tcl_FirstHashEntry(varPtr->value.tablePtr, &search); -		 hPtr != NULL; hPtr = Tcl_NextHashEntry(&search)) { -	        varPtr2 = (Var *) Tcl_GetHashValue(hPtr); -		if (TclIsVarUndefined(varPtr2)) { -		    continue; -		} -		name = Tcl_GetHashKey(varPtr->value.tablePtr, hPtr); -		if (objc > 3) { -		    switch ((enum options) mode) { -			case OPT_EXACT: -			    matched = (strcmp(name, pattern) == 0); -			    break; -			case OPT_GLOB: -			    matched = Tcl_StringMatch(name, pattern); -			    break; -			case OPT_REGEXP: -			    matched = Tcl_RegExpMatch(interp, name, -				    pattern); -			    if (matched < 0) { -				return TCL_ERROR; -			    } -			    break; -		    } -		    if (matched == 0) { -			continue; -		    } -		} -		 -		namePtr = Tcl_NewStringObj(name, -1); -		result = Tcl_ListObjAppendElement(interp, resultPtr, namePtr); -		if (result != TCL_OK) { -		    Tcl_DecrRefCount(namePtr); /* free unneeded name obj */ -		    return result; -		} -	    } -	    break; -	} -        case ARRAY_NEXTELEMENT: { -	    ArraySearch *searchPtr; -	    Tcl_HashEntry *hPtr; -	     -	    if (objc != 4) { -	        Tcl_WrongNumArgs(interp, 2, objv,  -                        "arrayName searchId"); -		return TCL_ERROR; -	    } -	    if (notArray) { -  	        goto error; -	    } -	    searchPtr = ParseSearchId(interp, varPtr, varName, objv[3]); -	    if (searchPtr == NULL) { -	        return TCL_ERROR; -	    } -	    while (1) { -	        Var *varPtr2; - -		hPtr = searchPtr->nextEntry; -		if (hPtr == NULL) { -		    hPtr = Tcl_NextHashEntry(&searchPtr->search); -		    if (hPtr == NULL) { -		        return TCL_OK; -		    } -		} else { -		    searchPtr->nextEntry = NULL; -		} -		varPtr2 = (Var *) Tcl_GetHashValue(hPtr); -		if (!TclIsVarUndefined(varPtr2)) { -		    break; -		} -	    } -	    Tcl_SetStringObj(resultPtr, -	            Tcl_GetHashKey(varPtr->value.tablePtr, hPtr), -1); -	    break; +	if (varPtr2 == protectedVarPtr) { +	    VarHashRefCount(varPtr2)--;  	} -        case ARRAY_SET: { -	    if (objc != 4) { -	        Tcl_WrongNumArgs(interp, 2, objv, "arrayName list"); -		return TCL_ERROR; -	    } -	    return(TclArraySet(interp, objv[2], objv[3])); -	} -        case ARRAY_SIZE: { -	    Tcl_HashSearch search; -	    Var *varPtr2; -	    int size; -	    if (objc != 3) { -	        Tcl_WrongNumArgs(interp, 2, objv, "arrayName"); -		return TCL_ERROR; -	    } -	    size = 0; -	    if (!notArray) { -	        for (hPtr = Tcl_FirstHashEntry(varPtr->value.tablePtr,  -                        &search); -		     hPtr != NULL;  hPtr = Tcl_NextHashEntry(&search)) { -		    varPtr2 = (Var *) Tcl_GetHashValue(hPtr); -		    if (TclIsVarUndefined(varPtr2)) { -		        continue; -		    } -		    size++; -		} -	    } -	    Tcl_SetIntObj(resultPtr, size); -	    break; +	/* +	 * Guard the next (peeked) item in the search chain by incrementing +	 * its refcount. This guarantees that the hash table iterator won't be +	 * dangling on the next time through the loop. +	 */ + +	if (search.nextEntryPtr != NULL) { +	    protectedVarPtr = VarHashGetValue(search.nextEntryPtr); +	    VarHashRefCount(protectedVarPtr)++; +	} else { +	    protectedVarPtr = NULL;  	} -        case ARRAY_STARTSEARCH: { -	    ArraySearch *searchPtr; -	    if (objc != 3) { -	        Tcl_WrongNumArgs(interp, 2, objv, "arrayName"); -		return TCL_ERROR; -	    } -	    if (notArray) { -	        goto error; -	    } -	    searchPtr = (ArraySearch *) ckalloc(sizeof(ArraySearch)); -	    if (varPtr->searchPtr == NULL) { -	        searchPtr->id = 1; -		Tcl_AppendStringsToObj(resultPtr, "s-1-", varName, -		        (char *) NULL); -	    } else { -	        char string[TCL_INTEGER_SPACE]; +	/* +	 * If the variable is undefined, clean it out as it has been hit by +	 * something else (i.e., an unset trace). +	 */ -		searchPtr->id = varPtr->searchPtr->id + 1; -		TclFormatInt(string, searchPtr->id); -		Tcl_AppendStringsToObj(resultPtr, "s-", string, "-", varName, -			(char *) NULL); -	    } -	    searchPtr->varPtr = varPtr; -	    searchPtr->nextEntry = Tcl_FirstHashEntry(varPtr->value.tablePtr, -		    &searchPtr->search); -	    searchPtr->nextPtr = varPtr->searchPtr; -	    varPtr->searchPtr = searchPtr; -	    break; +	if (TclIsVarUndefined(varPtr2)) { +	    CleanupVar(varPtr2, varPtr); +	    continue;  	} -	case ARRAY_STATISTICS: { -	    CONST char *stats; - -	    if (notArray) { -		goto error; -	    } +	nameObj = VarHashGetKey(varPtr2); +	if (Tcl_StringMatch(TclGetString(nameObj), pattern) +		&& TclPtrUnsetVar(interp, varPtr2, varPtr, varNameObj, +			nameObj, unsetFlags, -1) != TCL_OK) { +	    /* +	     * If we incremented a refcount, we must decrement it here as we +	     * will not be coming back properly due to the error. +	     */ -	    stats = Tcl_HashStats(varPtr->value.tablePtr); -	    if (stats != NULL) { -		Tcl_SetStringObj(Tcl_GetObjResult(interp), stats, -1); -		ckfree((void *)stats); -	    } else { -		Tcl_SetResult(interp, "error reading array statistics", -			TCL_STATIC); -		return TCL_ERROR; -	    } -	    break; -        } -	 -	case ARRAY_UNSET: { -	    Tcl_HashSearch search; -	    Var *varPtr2; -	    char *pattern = NULL; -	    char *name; -           -	    if ((objc != 3) && (objc != 4)) { -		Tcl_WrongNumArgs(interp, 2, objv, "arrayName ?pattern?"); -		return TCL_ERROR; -	    } -	    if (notArray) { -		return TCL_OK; -	    } -	    if (objc == 3) { -		/* -		 * When no pattern is given, just unset the whole array -		 */ -		if (Tcl_UnsetVar2(interp, varName, (char *) NULL, 0) -			!= TCL_OK) { -		    return TCL_ERROR; -		} -	    } else { -		pattern = Tcl_GetString(objv[3]); -		for (hPtr = Tcl_FirstHashEntry(varPtr->value.tablePtr, -			&search); -		     hPtr != NULL; hPtr = Tcl_NextHashEntry(&search)) { -		    varPtr2 = (Var *) Tcl_GetHashValue(hPtr); -		    if (TclIsVarUndefined(varPtr2)) { -			continue; -		    } -		    name = Tcl_GetHashKey(varPtr->value.tablePtr, hPtr); -		    if (Tcl_StringMatch(name, pattern) && -			    (Tcl_UnsetVar2(interp, varName, name, 0) -				    != TCL_OK)) { -			return TCL_ERROR; -		    } -		} +	    if (protectedVarPtr) { +		VarHashRefCount(protectedVarPtr)--; +		CleanupVar(protectedVarPtr, varPtr);  	    } -	    break; +	    return TCL_ERROR;  	}      }      return TCL_OK; - -    error: -    Tcl_AppendStringsToObj(resultPtr, "\"", varName, "\" isn't an array", -	    (char *) NULL); -    return TCL_ERROR;  }  /*   *----------------------------------------------------------------------   * - * TclArraySet -- + * TclInitArrayCmd --   * - *	Set the elements of an array.  If there are no elements to - *	set, create an empty array.  This routine is used by the - *	Tcl_ArrayObjCmd and by the TclSetupEnv routine. + *	This creates the ensemble for the "array" command.   *   * Results: - *	A standard Tcl result object. + *	The handle for the created ensemble.   *   * Side effects: - *	A variable will be created if one does not already exist. + *	Creates a command in the global namespace.   *   *----------------------------------------------------------------------   */ -int -TclArraySet(interp, arrayNameObj, arrayElemObj) -    Tcl_Interp *interp;		/* Current interpreter. */ -    Tcl_Obj *arrayNameObj;	/* The array name. */ -    Tcl_Obj *arrayElemObj;	/* The array elements list.  If this is -				 * NULL, create an empty array. */ +	/* ARGSUSED */ +Tcl_Command +TclInitArrayCmd( +    Tcl_Interp *interp)		/* Current interpreter. */  { -    Var *varPtr, *arrayPtr; -    Tcl_Obj **elemPtrs; -    int result, elemLen, i; -    char *varName, *p; -     -    varName = TclGetString(arrayNameObj); -    for (p = varName; *p ; p++) { -	if (*p == '(') { -	    do { -		p++; -	    } while (*p != '\0'); -	    p--; -	    if (*p == ')') { -		VarErrMsg(interp, varName, NULL, "set", needArray); -		return TCL_ERROR; -	    } -	    break; -	} -    } - -    varPtr = TclLookupVar(interp, varName, (char *) NULL, /*flags*/ 0, -            /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); - -    if (arrayElemObj != NULL) { -	result = Tcl_ListObjGetElements(interp, arrayElemObj, -		&elemLen, &elemPtrs); -	if (result != TCL_OK) { -	    return result; -	} -	if (elemLen & 1) { -	    Tcl_ResetResult(interp); -	    Tcl_AppendToObj(Tcl_GetObjResult(interp), -		    "list must have an even number of elements", -1); -	    return TCL_ERROR; -	} -	if (elemLen > 0) { -	    for (i = 0;  i < elemLen;  i += 2) { -		if (Tcl_ObjSetVar2(interp, arrayNameObj, elemPtrs[i], -			elemPtrs[i+1], TCL_LEAVE_ERR_MSG) == NULL) { -		    result = TCL_ERROR; -		    break; -		} -	    } -	    return result; -	} -    } -     -    /* -     * The list is empty make sure we have an array, or create -     * one if necessary. -     */ -     -    if (varPtr != NULL) { -	if (!TclIsVarUndefined(varPtr) && TclIsVarArray(varPtr)) { -	    /* -	     * Already an array, done. -	     */ -	     -	    return TCL_OK; -	} -	if (TclIsVarArrayElement(varPtr) || !TclIsVarUndefined(varPtr)) { -	    /* -	     * Either an array element, or a scalar: lose! -	     */ -	     -	    VarErrMsg(interp, varName, (char *)NULL, "array set", needArray); -	    return TCL_ERROR; -	} -    } else { -	/* -	 * Create variable for new array. -	 */ -	 -	varPtr = TclLookupVar(interp, varName, (char *) NULL, -		TCL_LEAVE_ERR_MSG, "set", -	        /*createPart1*/ 1, /*createPart2*/ 0, &arrayPtr); +    static const EnsembleImplMap arrayImplMap[] = { +	{"anymore",	ArrayAnyMoreCmd,	TclCompileBasic2ArgCmd, NULL, NULL, 0}, +	{"donesearch",	ArrayDoneSearchCmd,	TclCompileBasic2ArgCmd, NULL, NULL, 0}, +	{"exists",	ArrayExistsCmd,		TclCompileArrayExistsCmd, NULL, NULL, 0}, +	{"get",		ArrayGetCmd,		TclCompileBasic1Or2ArgCmd, NULL, NULL, 0}, +	{"names",	ArrayNamesCmd,		TclCompileBasic1To3ArgCmd, NULL, NULL, 0}, +	{"nextelement",	ArrayNextElementCmd,	TclCompileBasic2ArgCmd, NULL, NULL, 0}, +	{"set",		ArraySetCmd,		TclCompileArraySetCmd, NULL, NULL, 0}, +	{"size",	ArraySizeCmd,		TclCompileBasic1ArgCmd, NULL, NULL, 0}, +	{"startsearch",	ArrayStartSearchCmd,	TclCompileBasic1ArgCmd, NULL, NULL, 0}, +	{"statistics",	ArrayStatsCmd,		TclCompileBasic1ArgCmd, NULL, NULL, 0}, +	{"unset",	ArrayUnsetCmd,		TclCompileArrayUnsetCmd, NULL, NULL, 0}, +	{NULL, NULL, NULL, NULL, NULL, 0} +    }; -	/* -	 * Still couldn't do it - this can occur if a non-existent -	 * namespace was specified -	 */ -	if (varPtr == NULL) { -	    return TCL_ERROR; -	} -    } -    TclSetVarArray(varPtr); -    TclClearVarUndefined(varPtr); -    varPtr->value.tablePtr = -	(Tcl_HashTable *) ckalloc(sizeof(Tcl_HashTable)); -    Tcl_InitHashTable(varPtr->value.tablePtr, TCL_STRING_KEYS); -    return TCL_OK; +    return TclMakeEnsemble(interp, "array", arrayImplMap);  }  /*   *----------------------------------------------------------------------   * - * MakeUpvar -- + * ObjMakeUpvar --   * - *	This procedure does all of the work of the "global" and "upvar" + *	This function does all of the work of the "global" and "upvar"   *	commands.   *   * Results: - *	A standard Tcl completion code. If an error occurs then an - *	error message is left in iPtr->result. + *	A standard Tcl completion code. If an error occurs then an error + *	message is left in iPtr->result.   *   * Side effects:   *	The variable given by myName is linked to the variable in framePtr   *	given by otherP1 and otherP2, so that references to myName are   *	redirected to the other variable like a symbolic link. + *	Callers must Incr myNamePtr if they plan to Decr it. + *	Callers must Incr otherP1Ptr if they plan to Decr it.   *   *----------------------------------------------------------------------   */  static int -MakeUpvar(iPtr, framePtr, otherP1, otherP2, otherFlags, myName, myFlags) -    Interp *iPtr;		/* Interpreter containing variables. Used -				 * for error messages, too. */ -    CallFrame *framePtr;	/* Call frame containing "other" variable. +ObjMakeUpvar( +    Tcl_Interp *interp,		/* Interpreter containing variables. Used for +				 * error messages, too. */ +    CallFrame *framePtr,	/* Call frame containing "other" variable.  				 * NULL means use global :: context. */ -    char *otherP1, *otherP2;	/* Two-part name of variable in framePtr. */ -    int otherFlags;		/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: +    Tcl_Obj *otherP1Ptr, +    const char *otherP2,	/* Two-part name of variable in framePtr. */ +    const int otherFlags,	/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY:  				 * indicates scope of "other" variable. */ -    CONST char *myName;		/* Name of variable which will refer to +    Tcl_Obj *myNamePtr,		/* Name of variable which will refer to  				 * otherP1/otherP2. Must be a scalar. */ -    int myFlags;		/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: +    int myFlags,		/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY:  				 * indicates scope of myName. */ +    int index)			/* If the variable to be linked is an indexed +				 * scalar, this is its index. Otherwise, -1 */  { -    Tcl_HashEntry *hPtr; -    Var *otherPtr, *varPtr, *arrayPtr; +    Interp *iPtr = (Interp *) interp; +    Var *otherPtr, *arrayPtr;      CallFrame *varFramePtr; -    CallFrame *savedFramePtr = NULL;  /* Init. to avoid compiler warning. */ -    Tcl_HashTable *tablePtr; -    Namespace *nsPtr, *altNsPtr, *dummyNsPtr; -    CONST char *tail; -    int new;      /* -     * Find "other" in "framePtr". If not looking up other in just the -     * current namespace, temporarily replace the current var frame -     * pointer in the interpreter in order to use TclLookupVar. +     * Find "other" in "framePtr". If not looking up other in just the current +     * namespace, temporarily replace the current var frame pointer in the +     * interpreter in order to use TclObjLookupVar.       */ +    if (framePtr == NULL) { +	framePtr = iPtr->rootFramePtr; +    } + +    varFramePtr = iPtr->varFramePtr;      if (!(otherFlags & TCL_NAMESPACE_ONLY)) { -	savedFramePtr = iPtr->varFramePtr;  	iPtr->varFramePtr = framePtr;      } -    otherPtr = TclLookupVar((Tcl_Interp *) iPtr, otherP1, otherP2, +    otherPtr = TclObjLookupVar(interp, otherP1Ptr, otherP2,  	    (otherFlags | TCL_LEAVE_ERR_MSG), "access", -            /*createPart1*/ 1, /*createPart2*/ 1, &arrayPtr); +	    /*createPart1*/ 1, /*createPart2*/ 1, &arrayPtr);      if (!(otherFlags & TCL_NAMESPACE_ONLY)) { -	iPtr->varFramePtr = savedFramePtr; +	iPtr->varFramePtr = varFramePtr;      }      if (otherPtr == NULL) {  	return TCL_ERROR;      }      /* -     * Now create a hashtable entry for "myName". Create it as either a -     * namespace variable or as a local variable in a procedure call -     * frame. Interpret myName as a namespace variable if: -     *    1) so requested by a TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY flag, -     *    2) there is no active frame (we're at the global :: scope), -     *    3) the active frame was pushed to define the namespace context -     *       for a "namespace eval" or "namespace inscope" command, -     *    4) the name has namespace qualifiers ("::"s). -     * If creating myName in the active procedure, look first in the -     * frame's array of compiler-allocated local variables, then in its -     * hashtable for runtime-created local variables. Create that -     * procedure's local variable hashtable if necessary. +     * Check that we are not trying to create a namespace var linked to a +     * local variable in a procedure. If we allowed this, the local +     * variable in the shorter-lived procedure frame could go away leaving +     * the namespace var's reference invalid.       */ -    varFramePtr = iPtr->varFramePtr; -    if ((myFlags & (TCL_GLOBAL_ONLY | TCL_NAMESPACE_ONLY)) -	    || (varFramePtr == NULL) -	    || !varFramePtr->isProcCallFrame -	    || (strstr(myName, "::") != NULL)) { -	TclGetNamespaceForQualName((Tcl_Interp *) iPtr, myName, -		(Namespace *) NULL, myFlags, &nsPtr, &altNsPtr, &dummyNsPtr, &tail); - -        if (nsPtr == NULL) { -            nsPtr = altNsPtr; -        } -        if (nsPtr == NULL) { -	    Tcl_AppendResult((Tcl_Interp *) iPtr, "bad variable name \"", -		    myName, "\": unknown namespace", (char *) NULL); -            return TCL_ERROR; -        } -	 +    if (index < 0) { +	if (!(arrayPtr != NULL +		     ? (TclIsVarInHash(arrayPtr) && TclGetVarNsPtr(arrayPtr)) +		     : (TclIsVarInHash(otherPtr) && TclGetVarNsPtr(otherPtr))) +		&& ((myFlags & (TCL_GLOBAL_ONLY | TCL_NAMESPACE_ONLY)) +			|| (varFramePtr == NULL) +			|| !HasLocalVars(varFramePtr) +			|| (strstr(TclGetString(myNamePtr), "::") != NULL))) { +	    Tcl_SetObjResult((Tcl_Interp *) iPtr, Tcl_ObjPrintf( +		    "bad variable name \"%s\": upvar won't create " +		    "namespace variable that refers to procedure variable", +		    TclGetString(myNamePtr))); +	    Tcl_SetErrorCode(interp, "TCL", "UPVAR", "INVERTED", NULL); +	    return TCL_ERROR; +	} +    } + +    return TclPtrObjMakeUpvar(interp, otherPtr, myNamePtr, myFlags, index); +} + +/* + *---------------------------------------------------------------------- + * + * TclPtrMakeUpvar -- + * + *	This procedure does all of the work of the "global" and "upvar" + *	commands. + * + * Results: + *	A standard Tcl completion code. If an error occurs then an error + *	message is left in iPtr->result. + * + * Side effects: + *	The variable given by myName is linked to the variable in framePtr + *	given by otherP1 and otherP2, so that references to myName are + *	redirected to the other variable like a symbolic link. + * + *---------------------------------------------------------------------- + */ + +int +TclPtrMakeUpvar( +    Tcl_Interp *interp,		/* Interpreter containing variables. Used for +				 * error messages, too. */ +    Var *otherPtr,		/* Pointer to the variable being linked-to. */ +    const char *myName,		/* Name of variable which will refer to +				 * otherP1/otherP2. Must be a scalar. */ +    int myFlags,		/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: +				 * indicates scope of myName. */ +    int index)			/* If the variable to be linked is an indexed +				 * scalar, this is its index. Otherwise, -1 */ +{ +    Tcl_Obj *myNamePtr = NULL; +    int result; + +    if (myName) { +	myNamePtr = Tcl_NewStringObj(myName, -1); +	Tcl_IncrRefCount(myNamePtr); +    } +    result = TclPtrObjMakeUpvar(interp, otherPtr, myNamePtr, myFlags, index); +    if (myNamePtr) { +	Tcl_DecrRefCount(myNamePtr); +    } +    return result; +} + +/* Callers must Incr myNamePtr if they plan to Decr it. */ +  +int +TclPtrObjMakeUpvar( +    Tcl_Interp *interp,		/* Interpreter containing variables. Used for +				 * error messages, too. */ +    Var *otherPtr,		/* Pointer to the variable being linked-to. */ +    Tcl_Obj *myNamePtr,		/* Name of variable which will refer to +				 * otherP1/otherP2. Must be a scalar. */ +    int myFlags,		/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: +				 * indicates scope of myName. */ +    int index)			/* If the variable to be linked is an indexed +				 * scalar, this is its index. Otherwise, -1 */ +{ +    Interp *iPtr = (Interp *) interp; +    CallFrame *varFramePtr = iPtr->varFramePtr; +    const char *errMsg, *p, *myName; +    Var *varPtr; + +    if (index >= 0) { +	if (!HasLocalVars(varFramePtr)) { +	    Tcl_Panic("ObjMakeUpvar called with an index outside from a proc"); +	} +	varPtr = (Var *) &(varFramePtr->compiledLocals[index]); +	myNamePtr = localName(iPtr->varFramePtr, index); +	myName = myNamePtr? TclGetString(myNamePtr) : NULL; +    } else {  	/* -	 * Check that we are not trying to create a namespace var linked to -	 * a local variable in a procedure. If we allowed this, the local -	 * variable in the shorter-lived procedure frame could go away -	 * leaving the namespace var's reference invalid. +	 * Do not permit the new variable to look like an array reference, as +	 * it will not be reachable in that case [Bug 600812, TIP 184]. The +	 * "definition" of what "looks like an array reference" is consistent +	 * (and must remain consistent) with the code in TclObjLookupVar().  	 */ -	if ((otherP2 ? arrayPtr->nsPtr : otherPtr->nsPtr) == NULL) { -	    Tcl_AppendResult((Tcl_Interp *) iPtr, "bad variable name \"", -                    myName, "\": upvar won't create namespace variable that refers to procedure variable", -		    (char *) NULL); -            return TCL_ERROR; -        } -	 -	hPtr = Tcl_CreateHashEntry(&nsPtr->varTable, tail, &new); -	if (new) { -	    varPtr = NewVar(); -	    Tcl_SetHashValue(hPtr, varPtr); -	    varPtr->hPtr = hPtr; -            varPtr->nsPtr = nsPtr; -	} else { -	    varPtr = (Var *) Tcl_GetHashValue(hPtr); -	} -    } else {			/* look in the call frame */ -	Proc *procPtr = varFramePtr->procPtr; -	int localCt = procPtr->numCompiledLocals; -	CompiledLocal *localPtr = procPtr->firstLocalPtr; -	Var *localVarPtr = varFramePtr->compiledLocals; -	int nameLen = strlen(myName); -	int i; - -	varPtr = NULL; -	for (i = 0;  i < localCt;  i++) { -	    if (!TclIsVarTemporary(localPtr)) { -		char *localName = localVarPtr->name; -		if ((myName[0] == localName[0]) -		        && (nameLen == localPtr->nameLength) -		        && (strcmp(myName, localName) == 0)) { -		    varPtr = localVarPtr; -		    new = 0; -		    break; -		} +	myName = TclGetString(myNamePtr); +	p = strstr(myName, "("); +	if (p != NULL) { +	    p += strlen(p)-1; +	    if (*p == ')') { +		/* +		 * myName looks like an array reference. +		 */ + +		Tcl_SetObjResult((Tcl_Interp *) iPtr, Tcl_ObjPrintf( +			"bad variable name \"%s\": upvar won't create a" +			" scalar variable that looks like an array element", +			myName)); +		Tcl_SetErrorCode(interp, "TCL", "UPVAR", "LOCAL_ELEMENT", +			NULL); +		return TCL_ERROR;  	    } -	    localVarPtr++; -	    localPtr = localPtr->nextPtr;  	} -	if (varPtr == NULL) {	/* look in frame's local var hashtable */ -	    tablePtr = varFramePtr->varTablePtr; -	    if (tablePtr == NULL) { -		tablePtr = (Tcl_HashTable *) ckalloc(sizeof(Tcl_HashTable)); -		Tcl_InitHashTable(tablePtr, TCL_STRING_KEYS); -		varFramePtr->varTablePtr = tablePtr; -	    } -	    hPtr = Tcl_CreateHashEntry(tablePtr, myName, &new); -	    if (new) { -		varPtr = NewVar(); -		Tcl_SetHashValue(hPtr, varPtr); -		varPtr->hPtr = hPtr; -                varPtr->nsPtr = varFramePtr->nsPtr; -	    } else { -		varPtr = (Var *) Tcl_GetHashValue(hPtr); -	    } + +	/* +	 * Lookup and eventually create the new variable. Set the flag bit +	 * AVOID_RESOLVERS to indicate the special resolution rules for upvar +	 * purposes: +	 *   - Bug #696893 - variable is either proc-local or in the current +	 *     namespace; never follow the second (global) resolution path. +	 *   - Bug #631741 - do not use special namespace or interp resolvers. +	 */ + +	varPtr = TclLookupSimpleVar(interp, myNamePtr, +		myFlags|AVOID_RESOLVERS, /* create */ 1, &errMsg, &index); +	if (varPtr == NULL) { +	    TclObjVarErrMsg(interp, myNamePtr, NULL, "create", errMsg, -1); +	    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARNAME", +		    TclGetString(myNamePtr), NULL); +	    return TCL_ERROR;  	}      } -    if (!new) { +    if (varPtr == otherPtr) { +	Tcl_SetObjResult((Tcl_Interp *) iPtr, Tcl_NewStringObj( +		"can't upvar from variable to itself", -1)); +	Tcl_SetErrorCode(interp, "TCL", "UPVAR", "SELF", NULL); +	return TCL_ERROR; +    } + +    if (TclIsVarTraced(varPtr)) { +	Tcl_SetObjResult((Tcl_Interp *) iPtr, Tcl_ObjPrintf( +		"variable \"%s\" has traces: can't use for upvar", myName)); +	Tcl_SetErrorCode(interp, "TCL", "UPVAR", "TRACED", NULL); +	return TCL_ERROR; +    } else if (!TclIsVarUndefined(varPtr)) { +	Var *linkPtr; +  	/* -	 * The variable already exists. Make sure this variable "varPtr" -	 * isn't the same as "otherPtr" (avoid circular links). Also, if -	 * it's not an upvar then it's an error. If it is an upvar, then -	 * just disconnect it from the thing it currently refers to. +	 * The variable already existed. Make sure this variable "varPtr" +	 * isn't the same as "otherPtr" (avoid circular links). Also, if it's +	 * not an upvar then it's an error. If it is an upvar, then just +	 * disconnect it from the thing it currently refers to.  	 */ -	if (varPtr == otherPtr) { -	    Tcl_SetResult((Tcl_Interp *) iPtr, -		    "can't upvar from variable to itself", TCL_STATIC); +	if (!TclIsVarLink(varPtr)) { +	    Tcl_SetObjResult((Tcl_Interp *) iPtr, Tcl_ObjPrintf( +		    "variable \"%s\" already exists", myName)); +	    Tcl_SetErrorCode(interp, "TCL", "UPVAR", "EXISTS", NULL);  	    return TCL_ERROR;  	} -	if (TclIsVarLink(varPtr)) { -	    Var *linkPtr = varPtr->value.linkPtr; -	    if (linkPtr == otherPtr) { -		return TCL_OK; -	    } -	    linkPtr->refCount--; + +	linkPtr = varPtr->value.linkPtr; +	if (linkPtr == otherPtr) { +	    return TCL_OK; +	} +	if (TclIsVarInHash(linkPtr)) { +	    VarHashRefCount(linkPtr)--;  	    if (TclIsVarUndefined(linkPtr)) { -		CleanupVar(linkPtr, (Var *) NULL); +		CleanupVar(linkPtr, NULL);  	    } -	} else if (!TclIsVarUndefined(varPtr)) { -	    Tcl_AppendResult((Tcl_Interp *) iPtr, "variable \"", myName, -		    "\" already exists", (char *) NULL); -	    return TCL_ERROR; -	} else if (varPtr->tracePtr != NULL) { -	    Tcl_AppendResult((Tcl_Interp *) iPtr, "variable \"", myName, -		    "\" has traces: can't use for upvar", (char *) NULL); -	    return TCL_ERROR;  	}      }      TclSetVarLink(varPtr); -    TclClearVarUndefined(varPtr);      varPtr->value.linkPtr = otherPtr; -    otherPtr->refCount++; +    if (TclIsVarInHash(otherPtr)) { +	VarHashRefCount(otherPtr)++; +    }      return TCL_OK;  } @@ -4026,72 +4584,54 @@ MakeUpvar(iPtr, framePtr, otherP1, otherP2, otherFlags, myName, myFlags)   *   * Tcl_UpVar --   * - *	This procedure links one variable to another, just like - *	the "upvar" command. + *	This function links one variable to another, just like the "upvar" + *	command.   *   * Results: - *	A standard Tcl completion code.  If an error occurs then - *	an error message is left in the interp's result. + *	A standard Tcl completion code. If an error occurs then an error + *	message is left in the interp's result.   *   * Side effects:   *	The variable in frameName whose name is given by varName becomes - *	accessible under the name localName, so that references to - *	localName are redirected to the other variable like a symbolic + *	accessible under the name localNameStr, so that references to + *	localNameStr are redirected to the other variable like a symbolic   *	link.   *   *----------------------------------------------------------------------   */ +#undef Tcl_UpVar  int -Tcl_UpVar(interp, frameName, varName, localName, flags) -    Tcl_Interp *interp;		/* Command interpreter in which varName is -				 * to be looked up. */ -    CONST char *frameName;	/* Name of the frame containing the source +Tcl_UpVar( +    Tcl_Interp *interp,		/* Command interpreter in which varName is to +				 * be looked up. */ +    const char *frameName,	/* Name of the frame containing the source  				 * variable, such as "1" or "#0". */ -    char *varName;		/* Name of a variable in interp to link to. -				 * May be either a scalar name or an -				 * element in an array. */ -    CONST char *localName;	/* Name of link variable. */ -    int flags;			/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: -				 * indicates scope of localName. */ +    const char *varName,	/* Name of a variable in interp to link to. +				 * May be either a scalar name or an element +				 * in an array. */ +    const char *localNameStr,	/* Name of link variable. */ +    int flags)			/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: +				 * indicates scope of localNameStr. */  {      int result;      CallFrame *framePtr; -    register char *p; +    Tcl_Obj *varNamePtr, *localNamePtr; -    result = TclGetFrame(interp, frameName, &framePtr); -    if (result == -1) { +    if (TclGetFrame(interp, frameName, &framePtr) == -1) {  	return TCL_ERROR;      } -    /* -     * Figure out whether varName is an array reference, then call -     * MakeUpvar to do all the real work. -     */ - -    for (p = varName;  *p != '\0';  p++) { -	if (*p == '(') { -	    char *openParen = p; -	    do { -		p++; -	    } while (*p != '\0'); -	    p--; -	    if (*p != ')') { -		goto scalar; -	    } -	    *openParen = '\0'; -	    *p = '\0'; -	    result = MakeUpvar((Interp *) interp, framePtr, varName, -		    openParen+1, 0, localName, flags); -	    *openParen = '('; -	    *p = ')'; -	    return result; -	} -    } +    varNamePtr = Tcl_NewStringObj(varName, -1); +    Tcl_IncrRefCount(varNamePtr); +    localNamePtr = Tcl_NewStringObj(localNameStr, -1); +    Tcl_IncrRefCount(localNamePtr); -    scalar: -    return MakeUpvar((Interp *) interp, framePtr, varName, (char *) NULL, -	    0, localName, flags); +    result = ObjMakeUpvar(interp, framePtr, varNamePtr, NULL, 0, +	    localNamePtr, flags, -1); +    Tcl_DecrRefCount(varNamePtr); +    Tcl_DecrRefCount(localNamePtr); +    return result;  }  /* @@ -4099,43 +4639,53 @@ Tcl_UpVar(interp, frameName, varName, localName, flags)   *   * Tcl_UpVar2 --   * - *	This procedure links one variable to another, just like - *	the "upvar" command. + *	This function links one variable to another, just like the "upvar" + *	command.   *   * Results: - *	A standard Tcl completion code.  If an error occurs then - *	an error message is left in the interp's result. + *	A standard Tcl completion code. If an error occurs then an error + *	message is left in the interp's result.   *   * Side effects: - *	The variable in frameName whose name is given by part1 and - *	part2 becomes accessible under the name localName, so that - *	references to localName are redirected to the other variable - *	like a symbolic link. + *	The variable in frameName whose name is given by part1 and part2 + *	becomes accessible under the name localNameStr, so that references to + *	localNameStr are redirected to the other variable like a symbolic + *	link.   *   *----------------------------------------------------------------------   */  int -Tcl_UpVar2(interp, frameName, part1, part2, localName, flags) -    Tcl_Interp *interp;		/* Interpreter containing variables.  Used -				 * for error messages too. */ -    CONST char *frameName;	/* Name of the frame containing the source +Tcl_UpVar2( +    Tcl_Interp *interp,		/* Interpreter containing variables. Used for +				 * error messages too. */ +    const char *frameName,	/* Name of the frame containing the source  				 * variable, such as "1" or "#0". */ -    char *part1, *part2;	/* Two parts of source variable name to -				 * link to. */ -    CONST char *localName;	/* Name of link variable. */ -    int flags;			/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: -				 * indicates scope of localName. */ +    const char *part1, +    const char *part2,		/* Two parts of source variable name to link +				 * to. */ +    const char *localNameStr,	/* Name of link variable. */ +    int flags)			/* 0, TCL_GLOBAL_ONLY or TCL_NAMESPACE_ONLY: +				 * indicates scope of localNameStr. */  {      int result;      CallFrame *framePtr; +    Tcl_Obj *part1Ptr, *localNamePtr; -    result = TclGetFrame(interp, frameName, &framePtr); -    if (result == -1) { +    if (TclGetFrame(interp, frameName, &framePtr) == -1) {  	return TCL_ERROR;      } -    return MakeUpvar((Interp *) interp, framePtr, part1, part2, 0, -	    localName, flags); + +    part1Ptr = Tcl_NewStringObj(part1, -1); +    Tcl_IncrRefCount(part1Ptr); +    localNamePtr = Tcl_NewStringObj(localNameStr, -1); +    Tcl_IncrRefCount(localNamePtr); + +    result = ObjMakeUpvar(interp, framePtr, part1Ptr, part2, 0, +	    localNamePtr, flags, -1); +    Tcl_DecrRefCount(part1Ptr); +    Tcl_DecrRefCount(localNamePtr); +    return result;  }  /* @@ -4143,51 +4693,60 @@ Tcl_UpVar2(interp, frameName, part1, part2, localName, flags)   *   * Tcl_GetVariableFullName --   * - *	Given a Tcl_Var token returned by Tcl_FindNamespaceVar, this - *	procedure appends to an object the namespace variable's full - *	name, qualified by a sequence of parent namespace names. + *	Given a Tcl_Var token returned by Tcl_FindNamespaceVar, this function + *	appends to an object the namespace variable's full name, qualified by + *	a sequence of parent namespace names.   *   * Results: - *      None. + *	None.   *   * Side effects: - *      The variable's fully-qualified name is appended to the string + *	The variable's fully-qualified name is appended to the string   *	representation of objPtr.   *   *----------------------------------------------------------------------   */  void -Tcl_GetVariableFullName(interp, variable, objPtr) -    Tcl_Interp *interp;	        /* Interpreter containing the variable. */ -    Tcl_Var variable;		/* Token for the variable returned by a +Tcl_GetVariableFullName( +    Tcl_Interp *interp,		/* Interpreter containing the variable. */ +    Tcl_Var variable,		/* Token for the variable returned by a  				 * previous call to Tcl_FindNamespaceVar. */ -    Tcl_Obj *objPtr;		/* Points to the object onto which the +    Tcl_Obj *objPtr)		/* Points to the object onto which the  				 * variable's full name is appended. */  {      Interp *iPtr = (Interp *) interp;      register Var *varPtr = (Var *) variable; -    char *name; +    Tcl_Obj *namePtr; +    Namespace *nsPtr; + +    if (!varPtr || TclIsVarArrayElement(varPtr)) { +	return; +    }      /* -     * Add the full name of the containing namespace (if any), followed by -     * the "::" separator, then the variable name. +     * Add the full name of the containing namespace (if any), followed by the +     * "::" separator, then the variable name.       */ -    if (varPtr != NULL) { -	if (!TclIsVarArrayElement(varPtr)) { -	    if (varPtr->nsPtr != NULL) { -		Tcl_AppendToObj(objPtr, varPtr->nsPtr->fullName, -1); -		if (varPtr->nsPtr != iPtr->globalNsPtr) { -		    Tcl_AppendToObj(objPtr, "::", 2); -		} -	    } -	    if (varPtr->name != NULL) { -		Tcl_AppendToObj(objPtr, varPtr->name, -1); -	    } else if (varPtr->hPtr != NULL) { -		name = Tcl_GetHashKey(varPtr->hPtr->tablePtr, varPtr->hPtr); -		Tcl_AppendToObj(objPtr, name, -1); -	    } +    nsPtr = TclGetVarNsPtr(varPtr); +    if (nsPtr) { +	Tcl_AppendToObj(objPtr, nsPtr->fullName, -1); +	if (nsPtr != iPtr->globalNsPtr) { +	    Tcl_AppendToObj(objPtr, "::", 2); +	} +    } +    if (TclIsVarInHash(varPtr)) { +	if (!TclIsVarDeadHash(varPtr)) { +	    namePtr = VarHashGetKey(varPtr); +	    Tcl_AppendObjToObj(objPtr, namePtr); +	} +    } else if (iPtr->varFramePtr->procPtr) { +	int index = varPtr - iPtr->varFramePtr->compiledLocals; + +	if (index < iPtr->varFramePtr->numCompiledLocals) { +	    namePtr = localName(iPtr->varFramePtr, index); +	    Tcl_AppendObjToObj(objPtr, namePtr);  	}      }  } @@ -4197,7 +4756,7 @@ Tcl_GetVariableFullName(interp, variable, objPtr)   *   * Tcl_GlobalObjCmd --   * - *	This object-based procedure is invoked to process the "global" Tcl + *	This object-based function is invoked to process the "global" Tcl   *	command. See the user documentation for details on what it does.   *   * Results: @@ -4210,63 +4769,68 @@ Tcl_GetVariableFullName(interp, variable, objPtr)   */  int -Tcl_GlobalObjCmd(dummy, interp, objc, objv) -    ClientData dummy;		/* Not used. */ -    Tcl_Interp *interp;		/* Current interpreter. */ -    int objc;			/* Number of arguments. */ -    Tcl_Obj *CONST objv[];	/* Argument objects. */ +Tcl_GlobalObjCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  {      Interp *iPtr = (Interp *) interp; -    register Tcl_Obj *objPtr; -    char *varName; -    register char *tail; +    register Tcl_Obj *objPtr, *tailPtr; +    const char *varName; +    register const char *tail;      int result, i; -    if (objc < 2) { -	Tcl_WrongNumArgs(interp, 1, objv, "varName ?varName ...?"); -	return TCL_ERROR; -    } -      /*       * If we are not executing inside a Tcl procedure, just return.       */ -     -    if ((iPtr->varFramePtr == NULL) -	    || !iPtr->varFramePtr->isProcCallFrame) { + +    if (!HasLocalVars(iPtr->varFramePtr)) {  	return TCL_OK;      } -    for (i = 1;  i < objc;  i++) { +    for (i=1 ; i<objc ; i++) {  	/*  	 * Make a local variable linked to its counterpart in the global ::  	 * namespace.  	 */ -	 +  	objPtr = objv[i];  	varName = TclGetString(objPtr);  	/*  	 * The variable name might have a scope qualifier, but the name for -         * the local "link" variable must be the simple name at the tail. +	 * the local "link" variable must be the simple name at the tail.  	 */ -	for (tail = varName;  *tail != '\0';  tail++) { +	for (tail=varName ; *tail!='\0' ; tail++) {  	    /* empty body */  	} -        while ((tail > varName) && ((*tail != ':') || (*(tail-1) != ':'))) { -            tail--; +	while ((tail > varName) && ((*tail != ':') || (*(tail-1) != ':'))) { +	    tail--; +	} +	if ((*tail == ':') && (tail > varName)) { +	    tail++;  	} -        if ((*tail == ':') && (tail > varName)) { -            tail++; + +	if (tail == varName) { +	    tailPtr = objPtr; +	} else { +	    tailPtr = Tcl_NewStringObj(tail, -1); +	    Tcl_IncrRefCount(tailPtr);  	}  	/*  	 * Link to the variable "varName" in the global :: namespace.  	 */ -	 -	result = MakeUpvar(iPtr, (CallFrame *) NULL, -		varName, (char *) NULL, /*otherFlags*/ TCL_GLOBAL_ONLY, -	        /*myName*/ tail, /*myFlags*/ 0); + +	result = ObjMakeUpvar(interp, NULL, objPtr, NULL, +		TCL_GLOBAL_ONLY, /*myName*/ tailPtr, /*myFlags*/ 0, -1); + +	if (tail != varName) { +	    Tcl_DecrRefCount(tailPtr); +	} +  	if (result != TCL_OK) {  	    return result;  	} @@ -4289,102 +4853,96 @@ Tcl_GlobalObjCmd(dummy, interp, objc, objv)   *	optional.   *   *	If the variable does not exist, it is created and given the optional - *	value. If it already exists, it is simply set to the optional - *	value. Normally, "name" is an unqualified name, so it is created in - *	the current namespace. If it includes namespace qualifiers, it can - *	be created in another namespace. + *	value. If it already exists, it is simply set to the optional value. + *	Normally, "name" is an unqualified name, so it is created in the + *	current namespace. If it includes namespace qualifiers, it can be + *	created in another namespace.   * - *	If the variable command is executed inside a Tcl procedure, it - *	creates a local variable linked to the newly-created namespace - *	variable. + *	If the variable command is executed inside a Tcl procedure, it creates + *	a local variable linked to the newly-created namespace variable.   *   * Results: - *	Returns TCL_OK if the variable is found or created. Returns - *	TCL_ERROR if anything goes wrong. + *	Returns TCL_OK if the variable is found or created. Returns TCL_ERROR + *	if anything goes wrong.   *   * Side effects: - *	If anything goes wrong, this procedure returns an error message - *	as the result in the interpreter's result object. + *	If anything goes wrong, this function returns an error message as the + *	result in the interpreter's result object.   *   *----------------------------------------------------------------------   */  int -Tcl_VariableObjCmd(dummy, interp, objc, objv) -    ClientData dummy;		/* Not used. */ -    Tcl_Interp *interp;		/* Current interpreter. */ -    int objc;			/* Number of arguments. */ -    Tcl_Obj *CONST objv[];	/* Argument objects. */ +Tcl_VariableObjCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  {      Interp *iPtr = (Interp *) interp; -    char *varName, *tail, *cp; +    const char *varName, *tail, *cp;      Var *varPtr, *arrayPtr;      Tcl_Obj *varValuePtr;      int i, result; +    Tcl_Obj *varNamePtr, *tailPtr; -    if (objc < 2) { -	Tcl_WrongNumArgs(interp, 1, objv, "?name value...? name ?value?"); -	return TCL_ERROR; -    } - -    for (i = 1;  i < objc;  i = i+2) { +    for (i=1 ; i<objc ; i+=2) {  	/* -	 * Look up each variable in the current namespace context, creating -	 * it if necessary. +	 * Look up each variable in the current namespace context, creating it +	 * if necessary.  	 */ -	 -	varName = TclGetString(objv[i]); -	varPtr = TclLookupVar(interp, varName, (char *) NULL, -                (TCL_NAMESPACE_ONLY | TCL_LEAVE_ERR_MSG), "define", -                /*createPart1*/ 1, /*createPart2*/ 0, &arrayPtr); -	 -        if (arrayPtr != NULL) { -            /* -             * Variable cannot be an element in an array.  If arrayPtr is -             * non-null, it is, so throw up an error and return. -             */ -            VarErrMsg(interp, varName, NULL, "define", isArrayElement); -            return TCL_ERROR; -        } + +	varNamePtr = objv[i]; +	varName = TclGetString(varNamePtr); +	varPtr = TclObjLookupVarEx(interp, varNamePtr, NULL, +		(TCL_NAMESPACE_ONLY | TCL_LEAVE_ERR_MSG), "define", +		/*createPart1*/ 1, /*createPart2*/ 0, &arrayPtr); + +	if (arrayPtr != NULL) { +	    /* +	     * Variable cannot be an element in an array. If arrayPtr is +	     * non-NULL, it is, so throw up an error and return. +	     */ + +	    TclObjVarErrMsg(interp, varNamePtr, NULL, "define", +		    isArrayElement, -1); +	    Tcl_SetErrorCode(interp, "TCL", "UPVAR", "LOCAL_ELEMENT", NULL); +	    return TCL_ERROR; +	}  	if (varPtr == NULL) {  	    return TCL_ERROR;  	}  	/* -	 * Mark the variable as a namespace variable and increment its  +	 * Mark the variable as a namespace variable and increment its  	 * reference count so that it will persist until its namespace is  	 * destroyed or until the variable is unset.  	 */ -	if (!(varPtr->flags & VAR_NAMESPACE_VAR)) { -	    varPtr->flags |= VAR_NAMESPACE_VAR; -	    varPtr->refCount++; -	} +	TclSetVarNamespaceVar(varPtr);  	/*  	 * If a value was specified, set the variable to that value. -	 * Otherwise, if the variable is new, leave it undefined. -	 * (If the variable already exists and no value was specified, -	 * leave its value unchanged; just create the local link if -	 * we're in a Tcl procedure). +	 * Otherwise, if the variable is new, leave it undefined. (If the +	 * variable already exists and no value was specified, leave its value +	 * unchanged; just create the local link if we're in a Tcl procedure).  	 */ -	if (i+1 < objc) {	/* a value was specified */ -	    varValuePtr = Tcl_ObjSetVar2(interp, objv[i], NULL, objv[i+1], -		    (TCL_NAMESPACE_ONLY | TCL_LEAVE_ERR_MSG)); +	if (i+1 < objc) {	/* A value was specified. */ +	    varValuePtr = TclPtrSetVar(interp, varPtr, arrayPtr, varNamePtr, +		    NULL, objv[i+1], TCL_NAMESPACE_ONLY|TCL_LEAVE_ERR_MSG,-1);  	    if (varValuePtr == NULL) {  		return TCL_ERROR;  	    }  	}  	/* -	 * If we are executing inside a Tcl procedure, create a local -	 * variable linked to the new namespace variable "varName". +	 * If we are executing inside a Tcl procedure, create a local variable +	 * linked to the new namespace variable "varName".  	 */ -	if ((iPtr->varFramePtr != NULL) -	        && iPtr->varFramePtr->isProcCallFrame) { +	if (HasLocalVars(iPtr->varFramePtr)) {  	    /*  	     * varName might have a scope qualifier, but the name for the  	     * local "link" variable must be the simple name at the tail. @@ -4393,23 +4951,34 @@ Tcl_VariableObjCmd(dummy, interp, objc, objv)  	     * consecutive ":" characters).  	     */ -	    for (tail = cp = varName;  *cp != '\0'; ) { +	    for (tail=cp=varName ; *cp!='\0' ;) {  		if (*cp++ == ':') {  		    while (*cp == ':') {  			tail = ++cp;  		    }  		}  	    } -	     +  	    /*  	     * Create a local link "tail" to the variable "varName" in the  	     * current namespace.  	     */ -	     -	    result = MakeUpvar(iPtr, (CallFrame *) NULL, -		    /*otherP1*/ varName, /*otherP2*/ (char *) NULL, -                    /*otherFlags*/ TCL_NAMESPACE_ONLY, -		    /*myName*/ tail, /*myFlags*/ 0); + +	    if (tail == varName) { +		tailPtr = varNamePtr; +	    } else { +		tailPtr = Tcl_NewStringObj(tail, -1); +		Tcl_IncrRefCount(tailPtr); +	    } + +	    result = ObjMakeUpvar(interp, NULL, varNamePtr, /*otherP2*/ NULL, +		    /*otherFlags*/ TCL_NAMESPACE_ONLY, +		    /*myName*/ tailPtr, /*myFlags*/ 0, -1); + +	    if (tail != varName) { +		Tcl_DecrRefCount(tailPtr); +	    } +  	    if (result != TCL_OK) {  		return result;  	    } @@ -4423,8 +4992,8 @@ Tcl_VariableObjCmd(dummy, interp, objc, objv)   *   * Tcl_UpvarObjCmd --   * - *	This object-based procedure is invoked to process the "upvar" - *	Tcl command. See the user documentation for details on what it does. + *	This object-based function is invoked to process the "upvar" Tcl + *	command. See the user documentation for details on what it does.   *   * Results:   *	A standard Tcl object result value. @@ -4437,305 +5006,81 @@ Tcl_VariableObjCmd(dummy, interp, objc, objv)  	/* ARGSUSED */  int -Tcl_UpvarObjCmd(dummy, interp, objc, objv) -    ClientData dummy;		/* Not used. */ -    Tcl_Interp *interp;		/* Current interpreter. */ -    int objc;			/* Number of arguments. */ -    Tcl_Obj *CONST objv[];	/* Argument objects. */ +Tcl_UpvarObjCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  { -    register Interp *iPtr = (Interp *) interp;      CallFrame *framePtr; -    char *frameSpec, *otherVarName, *myVarName; -    register char *p; -    int result; +    int result, hasLevel; +    Tcl_Obj *levelObj;      if (objc < 3) { -	upvarSyntax:  	Tcl_WrongNumArgs(interp, 1, objv,  		"?level? otherVar localVar ?otherVar localVar ...?");  	return TCL_ERROR;      } -    /* -     * Find the call frame containing each of the "other variables" to be -     * linked to.  -     */ - -    frameSpec = TclGetString(objv[1]); -    result = TclGetFrame(interp, frameSpec, &framePtr); -    if (result == -1) { -	return TCL_ERROR; -    } -    objc -= result+1; -    if ((objc & 1) != 0) { -	goto upvarSyntax; -    } -    objv += result+1; - -    /* -     * Iterate over each (other variable, local variable) pair. -     * Divide the other variable name into two parts, then call -     * MakeUpvar to do all the work of linking it to the local variable. -     */ +    if (objc & 1) { +	/* +	 * Even number of arguments, so use the default level of "1" by +	 * passing NULL to TclObjGetFrame. +	 */ -    for ( ;  objc > 0;  objc -= 2, objv += 2) { -	myVarName = TclGetString(objv[1]); -	otherVarName = TclGetString(objv[0]); -	for (p = otherVarName;  *p != 0;  p++) { -	    if (*p == '(') { -		char *openParen = p; - -		do { -		    p++; -		} while (*p != '\0'); -		p--; -		if (*p != ')') { -		    goto scalar; -		} -		*openParen = '\0'; -		*p = '\0'; -		result = MakeUpvar(iPtr, framePtr, -		        otherVarName, openParen+1, /*otherFlags*/ 0, -			myVarName, /*flags*/ 0); -		*openParen = '('; -		*p = ')'; -		goto checkResult; -	    } -	} -	scalar: -	result = MakeUpvar(iPtr, framePtr, otherVarName, (char *) NULL, 0, -	        myVarName, /*flags*/ 0); +	levelObj = NULL; +	hasLevel = 0; +    } else { +	/* +	 * Odd number of arguments, so objv[1] must contain the level. +	 */ -	checkResult: -	if (result != TCL_OK) { -	    return TCL_ERROR; -	} +	levelObj = objv[1]; +	hasLevel = 1;      } -    return TCL_OK; -} - -/* - *---------------------------------------------------------------------- - * - * CallTraces -- - * - *	This procedure is invoked to find and invoke relevant - *	trace procedures associated with a particular operation on - *	a variable. This procedure invokes traces both on the - *	variable and on its containing array (where relevant). - * - * Results: - *	The return value is NULL if no trace procedures were invoked, or - *	if all the invoked trace procedures returned successfully. - *	The return value is non-NULL if a trace procedure returned an - *	error (in this case no more trace procedures were invoked after - *	the error was returned). In this case the return value is a - *	pointer to a static string describing the error. - * - * Side effects: - *	Almost anything can happen, depending on trace; this procedure - *	itself doesn't have any side effects. - * - *---------------------------------------------------------------------- - */ - -static char * -CallTraces(iPtr, arrayPtr, varPtr, part1, part2, flags, resultTypePtr) -    Interp *iPtr;		/* Interpreter containing variable. */ -    register Var *arrayPtr;	/* Pointer to array variable that contains -				 * the variable, or NULL if the variable -				 * isn't an element of an array. */ -    Var *varPtr;		/* Variable whose traces are to be -				 * invoked. */ -    char *part1, *part2;	/* Variable's two-part name. */ -    int flags;			/* Flags passed to trace procedures: -				 * indicates what's happening to variable, -				 * plus other stuff like TCL_GLOBAL_ONLY, -				 * TCL_NAMESPACE_ONLY, and -				 * TCL_INTERP_DESTROYED. */ -    int *resultTypePtr;		/* Report what kind of result was generated -				 * from the trace to this location. */ -{ -    register VarTrace *tracePtr; -    ActiveVarTrace active; -    char *result, *openParen, *p; -    Tcl_DString nameCopy; -    int copiedName;      /* -     * If there are already similar trace procedures active for the -     * variable, don't call them again. +     * Find the call frame containing each of the "other variables" to be +     * linked to.       */ -    if (varPtr->flags & VAR_TRACE_ACTIVE) { -	return NULL; -    } -    varPtr->flags |= VAR_TRACE_ACTIVE; -    varPtr->refCount++; -    if (arrayPtr != NULL) { -	arrayPtr->refCount++; +    result = TclObjGetFrame(interp, levelObj, &framePtr); +    if (result == -1) { +	return TCL_ERROR;      } +    if ((result == 0) && hasLevel) { +	/* +	 * Synthesize an error message since TclObjGetFrame doesn't do this +	 * for this particular case. +	 */ -    /* -     * If the variable name hasn't been parsed into array name and -     * element, do it here.  If there really is an array element, -     * make a copy of the original name so that NULLs can be -     * inserted into it to separate the names (can't modify the name -     * string in place, because the string might get used by the -     * callbacks we invoke). -     */ - -    copiedName = 0; -    if (part2 == NULL) { -	for (p = part1; *p ; p++) { -	    if (*p == '(') { -		openParen = p; -		do { -		    p++; -		} while (*p != '\0'); -		p--; -		if (*p == ')') { -		    Tcl_DStringInit(&nameCopy); -		    Tcl_DStringAppend(&nameCopy, part1, (p-part1)); -		    part2 = Tcl_DStringValue(&nameCopy) -			+ (openParen + 1 - part1); -		    part2[-1] = 0; -		    part1 = Tcl_DStringValue(&nameCopy); -		    copiedName = 1; -		} -		break; -	    } -	} +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"bad level \"%s\"", TclGetString(levelObj))); +	Tcl_SetErrorCode(interp, "TCL", "VALUE", "LEVEL", NULL); +	return TCL_ERROR;      }      /* -     * Invoke traces on the array containing the variable, if relevant. +     * We've now finished with parsing levels; skip to the variable names.       */ -    result = NULL; -    active.nextPtr = iPtr->activeTracePtr; -    iPtr->activeTracePtr = &active; -    Tcl_Preserve((ClientData) iPtr); -    if (arrayPtr != NULL && !(arrayPtr->flags & VAR_TRACE_ACTIVE)) { -	active.varPtr = arrayPtr; -	for (tracePtr = arrayPtr->tracePtr;  tracePtr != NULL; -	     tracePtr = active.nextTracePtr) { -	    active.nextTracePtr = tracePtr->nextPtr; -	    if (!(tracePtr->flags & flags)) { -		continue; -	    } -	    Tcl_Preserve((ClientData) tracePtr); -	    result = (*tracePtr->traceProc)(tracePtr->clientData, -		    (Tcl_Interp *) iPtr, part1, part2, flags); -	    if (result != NULL) { -		*resultTypePtr = tracePtr->flags & -			(TCL_TRACE_RESULT_DYNAMIC | TCL_TRACE_RESULT_OBJECT); -		if (flags & TCL_TRACE_UNSETS) { -		    if (tracePtr->flags & TCL_TRACE_RESULT_DYNAMIC) { -			ckfree(result); -		    } else if (tracePtr->flags & TCL_TRACE_RESULT_OBJECT) { -			Tcl_DecrRefCount((Tcl_Obj *) result); -		    } -		    result = NULL; -		} else { -	            Tcl_Release((ClientData) tracePtr); -		    goto done; -		} -	    } -	    Tcl_Release((ClientData) tracePtr); -	} -    } +    objc -= hasLevel + 1; +    objv += hasLevel + 1;      /* -     * Invoke traces on the variable itself. +     * Iterate over each (other variable, local variable) pair. Divide the +     * other variable name into two parts, then call MakeUpvar to do all the +     * work of linking it to the local variable.       */ -    if (flags & TCL_TRACE_UNSETS) { -	flags |= TCL_TRACE_DESTROYED; -    } -    active.varPtr = varPtr; -    for (tracePtr = varPtr->tracePtr; tracePtr != NULL; -	 tracePtr = active.nextTracePtr) { -	active.nextTracePtr = tracePtr->nextPtr; -	if (!(tracePtr->flags & flags)) { -	    continue; -	} -	Tcl_Preserve((ClientData) tracePtr); -	result = (*tracePtr->traceProc)(tracePtr->clientData, -		(Tcl_Interp *) iPtr, part1, part2, flags); -	if (result != NULL) { -	    *resultTypePtr = tracePtr->flags & -		    (TCL_TRACE_RESULT_DYNAMIC | TCL_TRACE_RESULT_OBJECT); -	    if (flags & TCL_TRACE_UNSETS) { -		if (tracePtr->flags & TCL_TRACE_RESULT_DYNAMIC) { -		    ckfree(result); -		} else if (tracePtr->flags & TCL_TRACE_RESULT_OBJECT) { -		    Tcl_DecrRefCount((Tcl_Obj *) result); -		} -		result = NULL; -	    } else { -	        Tcl_Release((ClientData) tracePtr); -		goto done; -	    } +    for (; objc>0 ; objc-=2, objv+=2) { +	result = ObjMakeUpvar(interp, framePtr, /* othervarName */ objv[0], +		NULL, 0, /* myVarName */ objv[1], /*flags*/ 0, -1); +	if (result != TCL_OK) { +	    return TCL_ERROR;  	} -	Tcl_Release((ClientData) tracePtr); -    } - -    /* -     * Restore the variable's flags, remove the record of our active -     * traces, and then return. -     */ - -    done: -    if (arrayPtr != NULL) { -	arrayPtr->refCount--; -    } -    if (copiedName) { -	Tcl_DStringFree(&nameCopy);      } -    varPtr->flags &= ~VAR_TRACE_ACTIVE; -    varPtr->refCount--; -    iPtr->activeTracePtr = active.nextPtr; -    Tcl_Release((ClientData) iPtr); -    return result; -} - -/* - *---------------------------------------------------------------------- - * - * NewVar -- - * - *	Create a new heap-allocated variable that will eventually be - *	entered into a hashtable. - * - * Results: - *	The return value is a pointer to the new variable structure. It is - *	marked as a scalar variable (and not a link or array variable). Its - *	value initially is NULL. The variable is not part of any hash table - *	yet. Since it will be in a hashtable and not in a call frame, its - *	name field is set NULL. It is initially marked as undefined. - * - * Side effects: - *	Storage gets allocated. - * - *---------------------------------------------------------------------- - */ - -static Var * -NewVar() -{ -    register Var *varPtr; - -    varPtr = (Var *) ckalloc(sizeof(Var)); -    varPtr->value.objPtr = NULL; -    varPtr->name = NULL; -    varPtr->nsPtr = NULL; -    varPtr->hPtr = NULL; -    varPtr->refCount = 0; -    varPtr->tracePtr = NULL; -    varPtr->searchPtr = NULL; -    varPtr->flags = (VAR_SCALAR | VAR_UNDEFINED | VAR_IN_HASHTABLE); -    return varPtr; +    return TCL_OK;  }  /* @@ -4743,29 +5088,28 @@ NewVar()   *   * SetArraySearchObj --   * - *	This function converts the given tcl object into one that - *	has the "array search" internal type. + *	This function converts the given tcl object into one that has the + *	"array search" internal type.   *   * Results: - *	TCL_OK if the conversion succeeded, and TCL_ERROR if it failed - *	(when an error message will be placed in the interpreter's - *	result.) + *	TCL_OK if the conversion succeeded, and TCL_ERROR if it failed (when + *	an error message will be placed in the interpreter's result.)   *   * Side effects: - *	Updates the internal type and representation of the object to - *	make this an array-search object.  See the tclArraySearchType - *	declaration above for details of the internal representation. + *	Updates the internal type and representation of the object to make + *	this an array-search object. See the tclArraySearchType declaration + *	above for details of the internal representation.   *   *----------------------------------------------------------------------   */  static int -SetArraySearchObj(interp, objPtr) -    Tcl_Interp *interp; -    Tcl_Obj *objPtr; +SetArraySearchObj( +    Tcl_Interp *interp, +    Tcl_Obj *objPtr)  { -    char *string; -    char *end; +    const char *string; +    char *end;			/* Can't be const due to strtoul defn. */      int id;      size_t offset; @@ -4773,35 +5117,39 @@ SetArraySearchObj(interp, objPtr)       * Get the string representation. Make it up-to-date if necessary.       */ -    string = Tcl_GetString(objPtr); +    string = TclGetString(objPtr);      /*       * Parse the id into the three parts separated by dashes.       */ +      if ((string[0] != 's') || (string[1] != '-')) { -	syntax: -	Tcl_AppendResult(interp, "illegal search identifier \"", string, -		"\"", (char *) NULL); -	return TCL_ERROR; +	goto syntax;      }      id = strtoul(string+2, &end, 10);      if ((end == (string+2)) || (*end != '-')) {  	goto syntax;      } +      /* -     * Can't perform value check in this context, so place reference -     * to place in string to use for the check in the object instead. +     * Can't perform value check in this context, so place reference to place +     * in string to use for the check in the object instead.       */ +      end++;      offset = end - string; -    if (objPtr->typePtr != NULL && objPtr->typePtr->freeIntRepProc != NULL) { -	objPtr->typePtr->freeIntRepProc(objPtr); -    } +    TclFreeIntRep(objPtr);      objPtr->typePtr = &tclArraySearchType; -    objPtr->internalRep.twoPtrValue.ptr1 = (VOID *)(((char *)NULL)+id); -    objPtr->internalRep.twoPtrValue.ptr2 = (VOID *)(((char *)NULL)+offset); +    objPtr->internalRep.twoPtrValue.ptr1 = INT2PTR(id); +    objPtr->internalRep.twoPtrValue.ptr2 = INT2PTR(offset);      return TCL_OK; + +  syntax: +    Tcl_SetObjResult(interp, Tcl_ObjPrintf( +	    "illegal search identifier \"%s\"", string)); +    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ARRAYSEARCH", string, NULL); +    return TCL_ERROR;  }  /* @@ -4809,13 +5157,13 @@ SetArraySearchObj(interp, objPtr)   *   * ParseSearchId --   * - *	This procedure translates from a tcl object to a pointer to an - *	active array search (if there is one that matches the string). + *	This function translates from a tcl object to a pointer to an active + *	array search (if there is one that matches the string).   *   * Results: - *	The return value is a pointer to the array search indicated - *	by string, or NULL if there isn't one.  If NULL is returned, - *	the interp's result contains an error message. + *	The return value is a pointer to the array search indicated by string, + *	or NULL if there isn't one. If NULL is returned, the interp's result + *	contains an error message.   *   * Side effects:   *	The tcl object might have its internal type and representation @@ -4825,62 +5173,76 @@ SetArraySearchObj(interp, objPtr)   */  static ArraySearch * -ParseSearchId(interp, varPtr, varName, handleObj) -    Tcl_Interp *interp;		/* Interpreter containing variable. */ -    Var *varPtr;		/* Array variable search is for. */ -    char *varName;		/* Name of array variable that search is +ParseSearchId( +    Tcl_Interp *interp,		/* Interpreter containing variable. */ +    const Var *varPtr,		/* Array variable search is for. */ +    Tcl_Obj *varNamePtr,	/* Name of array variable that search is  				 * supposed to be for. */ -    Tcl_Obj *handleObj;		/* Object containing id of search. Must have +    Tcl_Obj *handleObj)		/* Object containing id of search. Must have  				 * form "search-num-var" where "num" is a  				 * decimal number and "var" is a variable  				 * name. */  { -    register char *string; +    Interp *iPtr = (Interp *) interp; +    register const char *string;      register size_t offset;      int id;      ArraySearch *searchPtr; +    const char *varName = TclGetString(varNamePtr);      /*       * Parse the id.       */ -    if (Tcl_ConvertToType(interp, handleObj, &tclArraySearchType) != TCL_OK) { + +    if ((handleObj->typePtr != &tclArraySearchType) +	    && (SetArraySearchObj(interp, handleObj) != TCL_OK)) {  	return NULL;      } +      /* -     * Cast is safe, since always came from an int in the first place. +     * Extract the information out of the Tcl_Obj.       */ -    id = (int)(((char*)handleObj->internalRep.twoPtrValue.ptr1) - -	       ((char*)NULL)); -    string = Tcl_GetString(handleObj); -    offset = (((char*)handleObj->internalRep.twoPtrValue.ptr2) - -	      ((char*)NULL)); + +    id = PTR2INT(handleObj->internalRep.twoPtrValue.ptr1); +    string = TclGetString(handleObj); +    offset = PTR2INT(handleObj->internalRep.twoPtrValue.ptr2); +      /* -     * This test cannot be placed inside the Tcl_Obj machinery, since -     * it is dependent on the variable context. +     * This test cannot be placed inside the Tcl_Obj machinery, since it is +     * dependent on the variable context.       */ +      if (strcmp(string+offset, varName) != 0) { -	Tcl_AppendResult(interp, "search identifier \"", string, -		"\" isn't for variable \"", varName, "\"", (char *) NULL); -	return NULL; +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"search identifier \"%s\" isn't for variable \"%s\"", +		string, varName)); +	goto badLookup;      }      /* -     * Search through the list of active searches on the interpreter -     * to see if the desired one exists. +     * Search through the list of active searches on the interpreter to see if +     * the desired one exists.       * -     * Note that we cannot store the searchPtr directly in the Tcl_Obj -     * as that would run into trouble when DeleteSearches() was called -     * so we must scan this list every time. +     * Note that we cannot store the searchPtr directly in the Tcl_Obj as that +     * would run into trouble when DeleteSearches() was called so we must scan +     * this list every time.       */ -    for (searchPtr = varPtr->searchPtr; searchPtr != NULL; -	 searchPtr = searchPtr->nextPtr) { -	if (searchPtr->id == id) { -	    return searchPtr; +    if (varPtr->flags & VAR_SEARCH_ACTIVE) { +	Tcl_HashEntry *hPtr = +		Tcl_FindHashEntry(&iPtr->varSearches, varPtr); + +	for (searchPtr = Tcl_GetHashValue(hPtr); searchPtr != NULL; +		searchPtr = searchPtr->nextPtr) { +	    if (searchPtr->id == id) { +		return searchPtr; +	    }  	}      } -    Tcl_AppendResult(interp, "couldn't find search \"", string, "\"", -	    (char *) NULL); +    Tcl_SetObjResult(interp, Tcl_ObjPrintf( +	    "couldn't find search \"%s\"", string)); +  badLookup: +    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ARRAYSEARCH", string, NULL);      return NULL;  } @@ -4889,8 +5251,8 @@ ParseSearchId(interp, varPtr, varName, handleObj)   *   * DeleteSearches --   * - *	This procedure is called to free up all of the searches - *	associated with an array variable. + *	This function is called to free up all of the searches associated + *	with an array variable.   *   * Results:   *	None. @@ -4902,17 +5264,105 @@ ParseSearchId(interp, varPtr, varName, handleObj)   */  static void -DeleteSearches(arrayVarPtr) -    register Var *arrayVarPtr;		/* Variable whose searches are -					 * to be deleted. */ +DeleteSearches( +    Interp *iPtr, +    register Var *arrayVarPtr)	/* Variable whose searches are to be +				 * deleted. */  { -    ArraySearch *searchPtr; +    ArraySearch *searchPtr, *nextPtr; +    Tcl_HashEntry *sPtr; + +    if (arrayVarPtr->flags & VAR_SEARCH_ACTIVE) { +	sPtr = Tcl_FindHashEntry(&iPtr->varSearches, arrayVarPtr); +	for (searchPtr = Tcl_GetHashValue(sPtr); searchPtr != NULL; +		searchPtr = nextPtr) { +	    nextPtr = searchPtr->nextPtr; +	    ckfree(searchPtr); +	} +	arrayVarPtr->flags &= ~VAR_SEARCH_ACTIVE; +	Tcl_DeleteHashEntry(sPtr); +    } +} + +/* + *---------------------------------------------------------------------- + * + * TclDeleteNamespaceVars -- + * + *	This function is called to recycle all the storage space associated + *	with a namespace's table of variables. + * + * Results: + *	None. + * + * Side effects: + *	Variables are deleted and trace functions are invoked, if any are + *	declared. + * + *---------------------------------------------------------------------- + */ + +void +TclDeleteNamespaceVars( +    Namespace *nsPtr) +{ +    TclVarHashTable *tablePtr = &nsPtr->varTable; +    Tcl_Interp *interp = nsPtr->interp; +    Interp *iPtr = (Interp *)interp; +    Tcl_HashSearch search; +    int flags = 0; +    Var *varPtr; + +    /* +     * Determine what flags to pass to the trace callback functions. +     */ + +    if (nsPtr == iPtr->globalNsPtr) { +	flags = TCL_GLOBAL_ONLY; +    } else if (nsPtr == (Namespace *) TclGetCurrentNamespace(interp)) { +	flags = TCL_NAMESPACE_ONLY; +    } + +    for (varPtr = VarHashFirstVar(tablePtr, &search);  varPtr != NULL; +	    varPtr = VarHashFirstVar(tablePtr, &search)) { +	Tcl_Obj *objPtr = Tcl_NewObj(); +	VarHashRefCount(varPtr)++;	/* Make sure we get to remove from +					 * hash. */ +	Tcl_GetVariableFullName(interp, (Tcl_Var) varPtr, objPtr); +	UnsetVarStruct(varPtr, NULL, iPtr, /* part1 */ objPtr, NULL, flags, +		-1); +	Tcl_DecrRefCount(objPtr);	/* Free no longer needed obj */ + +	/* +	 * Remove the variable from the table and force it undefined in case +	 * an unset trace brought it back from the dead. +	 */ + +	if (TclIsVarTraced(varPtr)) { +	    Tcl_HashEntry *tPtr = Tcl_FindHashEntry(&iPtr->varTraces, varPtr); +	    VarTrace *tracePtr = Tcl_GetHashValue(tPtr); +	    ActiveVarTrace *activePtr; -    while (arrayVarPtr->searchPtr != NULL) { -	searchPtr = arrayVarPtr->searchPtr; -	arrayVarPtr->searchPtr = searchPtr->nextPtr; -	ckfree((char *) searchPtr); +	    while (tracePtr) { +		VarTrace *prevPtr = tracePtr; + +		tracePtr = tracePtr->nextPtr; +		prevPtr->nextPtr = NULL; +		Tcl_EventuallyFree(prevPtr, TCL_DYNAMIC); +	    } +	    Tcl_DeleteHashEntry(tPtr); +	    varPtr->flags &= ~VAR_ALL_TRACES; +	    for (activePtr = iPtr->activeVarTracePtr; activePtr != NULL; +		    activePtr = activePtr->nextPtr) { +		if (activePtr->varPtr == varPtr) { +		    activePtr->nextTracePtr = NULL; +		} +	    } +	} +	VarHashRefCount(varPtr)--; +	VarHashDeleteEntry(varPtr);      } +    VarHashDeleteTable(tablePtr);  }  /* @@ -4920,513 +5370,1189 @@ DeleteSearches(arrayVarPtr)   *   * TclDeleteVars --   * - *	This procedure is called to recycle all the storage space - *	associated with a table of variables. For this procedure - *	to work correctly, it must not be possible for any of the - *	variables in the table to be accessed from Tcl commands - *	(e.g. from trace procedures). + *	This function is called to recycle all the storage space associated + *	with a table of variables. For this function to work correctly, it + *	must not be possible for any of the variables in the table to be + *	accessed from Tcl commands (e.g. from trace functions).   *   * Results:   *	None.   *   * Side effects: - *	Variables are deleted and trace procedures are invoked, if - *	any are declared. + *	Variables are deleted and trace functions are invoked, if any are + *	declared.   *   *----------------------------------------------------------------------   */  void -TclDeleteVars(iPtr, tablePtr) -    Interp *iPtr;		/* Interpreter to which variables belong. */ -    Tcl_HashTable *tablePtr;	/* Hash table containing variables to +TclDeleteVars( +    Interp *iPtr,		/* Interpreter to which variables belong. */ +    TclVarHashTable *tablePtr)	/* Hash table containing variables to  				 * delete. */  {      Tcl_Interp *interp = (Tcl_Interp *) iPtr;      Tcl_HashSearch search; -    Tcl_HashEntry *hPtr;      register Var *varPtr; -    Var *linkPtr;      int flags; -    ActiveVarTrace *activePtr; -    Tcl_Obj *objPtr; -    Namespace *currNsPtr = (Namespace *) Tcl_GetCurrentNamespace(interp); +    Namespace *currNsPtr = (Namespace *) TclGetCurrentNamespace(interp);      /* -     * Determine what flags to pass to the trace callback procedures. +     * Determine what flags to pass to the trace callback functions.       */      flags = TCL_TRACE_UNSETS;      if (tablePtr == &iPtr->globalNsPtr->varTable) { -	flags |= (TCL_INTERP_DESTROYED | TCL_GLOBAL_ONLY); +	flags |= TCL_GLOBAL_ONLY;      } else if (tablePtr == &currNsPtr->varTable) {  	flags |= TCL_NAMESPACE_ONLY;      } -    for (hPtr = Tcl_FirstHashEntry(tablePtr, &search);  hPtr != NULL; -	 hPtr = Tcl_NextHashEntry(&search)) { -	varPtr = (Var *) Tcl_GetHashValue(hPtr); +    for (varPtr = VarHashFirstVar(tablePtr, &search); varPtr != NULL; +	 varPtr = VarHashFirstVar(tablePtr, &search)) { +	UnsetVarStruct(varPtr, NULL, iPtr, VarHashGetKey(varPtr), NULL, flags, +		-1); +	VarHashDeleteEntry(varPtr); +    } +    VarHashDeleteTable(tablePtr); +} + +/* + *---------------------------------------------------------------------- + * + * TclDeleteCompiledLocalVars -- + * + *	This function is called to recycle storage space associated with the + *	compiler-allocated array of local variables in a procedure call frame. + *	This function resembles TclDeleteVars above except that each variable + *	is stored in a call frame and not a hash table. For this function to + *	work correctly, it must not be possible for any of the variable in the + *	table to be accessed from Tcl commands (e.g. from trace functions). + * + * Results: + *	None. + * + * Side effects: + *	Variables are deleted and trace functions are invoked, if any are + *	declared. + * + *---------------------------------------------------------------------- + */ -	/* -	 * For global/upvar variables referenced in procedures, decrement -	 * the reference count on the variable referred to, and free -	 * the referenced variable if it's no longer needed. Don't delete -	 * the hash entry for the other variable if it's in the same table -	 * as us: this will happen automatically later on. -	 */ +void +TclDeleteCompiledLocalVars( +    Interp *iPtr,		/* Interpreter to which variables belong. */ +    CallFrame *framePtr)	/* Procedure call frame containing compiler- +				 * assigned local variables to delete. */ +{ +    register Var *varPtr; +    int numLocals, i; +    Tcl_Obj **namePtrPtr; -	if (TclIsVarLink(varPtr)) { -	    linkPtr = varPtr->value.linkPtr; -	    linkPtr->refCount--; -	    if ((linkPtr->refCount == 0) && TclIsVarUndefined(linkPtr) -		    && (linkPtr->tracePtr == NULL) -		    && (linkPtr->flags & VAR_IN_HASHTABLE)) { -		if (linkPtr->hPtr == NULL) { -		    ckfree((char *) linkPtr); -		} else if (linkPtr->hPtr->tablePtr != tablePtr) { -		    Tcl_DeleteHashEntry(linkPtr->hPtr); -		    ckfree((char *) linkPtr); -		} -	    } +    numLocals = framePtr->numCompiledLocals; +    varPtr = framePtr->compiledLocals; +    namePtrPtr = &localName(framePtr, 0); +    for (i=0 ; i<numLocals ; i++, namePtrPtr++, varPtr++) { +	UnsetVarStruct(varPtr, NULL, iPtr, *namePtrPtr, NULL, +		TCL_TRACE_UNSETS, i); +    } +    framePtr->numCompiledLocals = 0; +} + +/* + *---------------------------------------------------------------------- + * + * DeleteArray -- + * + *	This function is called to free up everything in an array variable. + *	It's the caller's responsibility to make sure that the array is no + *	longer accessible before this function is called. + * + * Results: + *	None. + * + * Side effects: + *	All storage associated with varPtr's array elements is deleted + *	(including the array's hash table). Deletion trace functions for + *	array elements are invoked, then deleted. Any pending traces for array + *	elements are also deleted. + * + *---------------------------------------------------------------------- + */ + +static void +DeleteArray( +    Interp *iPtr,		/* Interpreter containing array. */ +    Tcl_Obj *arrayNamePtr,	/* Name of array (used for trace callbacks), +				 * or NULL if it is to be computed on +				 * demand. */ +    Var *varPtr,		/* Pointer to variable structure. */ +    int flags,			/* Flags to pass to TclCallVarTraces: +				 * TCL_TRACE_UNSETS and sometimes +				 * TCL_NAMESPACE_ONLY or TCL_GLOBAL_ONLY. */ +    int index) +{ +    Tcl_HashSearch search; +    Tcl_HashEntry *tPtr; +    register Var *elPtr; +    ActiveVarTrace *activePtr; +    Tcl_Obj *objPtr; +    VarTrace *tracePtr; + +    if (varPtr->flags & VAR_SEARCH_ACTIVE) { +	DeleteSearches(iPtr, varPtr); +    } +    for (elPtr = VarHashFirstVar(varPtr->value.tablePtr, &search); +	    elPtr != NULL; elPtr = VarHashNextVar(&search)) { +	if (TclIsVarScalar(elPtr) && (elPtr->value.objPtr != NULL)) { +	    objPtr = elPtr->value.objPtr; +	    TclDecrRefCount(objPtr); +	    elPtr->value.objPtr = NULL;  	}  	/* -	 * Invoke traces on the variable that is being deleted, then -	 * free up the variable's space (no need to free the hash entry -	 * here, unless we're dealing with a global variable: the -	 * hash entries will be deleted automatically when the whole -	 * table is deleted). Note that we give CallTraces the variable's -	 * fully-qualified name so that any called trace procedures can -	 * refer to these variables being deleted. +	 * Lie about the validity of the hashtable entry. In this way the +	 * variables will be deleted by VarHashDeleteTable.  	 */ -	if (varPtr->tracePtr != NULL) { -	    char *msg; -	    int resultType; - -	    objPtr = Tcl_NewObj(); -	    Tcl_IncrRefCount(objPtr); /* until done with traces */ -	    Tcl_GetVariableFullName(interp, (Tcl_Var) varPtr, objPtr); -	    msg = CallTraces(iPtr, (Var *) NULL, varPtr, -		    Tcl_GetString(objPtr), (char *) NULL, flags, &resultType); -	    if (msg != NULL) { -		if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		    ckfree(msg); -		} else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    Tcl_DecrRefCount((Tcl_Obj *) msg); -		} +	VarHashInvalidateEntry(elPtr); +	if (TclIsVarTraced(elPtr)) { +	    /* +	     * Compute the array name if it was not supplied. +	     */ + +	    if (elPtr->flags & VAR_TRACED_UNSET) { +		Tcl_Obj *elNamePtr = VarHashGetKey(elPtr); + +		elPtr->flags &= ~VAR_TRACE_ACTIVE; +		TclObjCallVarTraces(iPtr, NULL, elPtr, arrayNamePtr, +			elNamePtr, flags,/* leaveErrMsg */ 0, index);  	    } -	    Tcl_DecrRefCount(objPtr); /* free no longer needed obj */ +	    tPtr = Tcl_FindHashEntry(&iPtr->varTraces, elPtr); +	    tracePtr = Tcl_GetHashValue(tPtr); +	    while (tracePtr) { +		VarTrace *prevPtr = tracePtr; -	    while (varPtr->tracePtr != NULL) { -		VarTrace *tracePtr = varPtr->tracePtr; -		varPtr->tracePtr = tracePtr->nextPtr; -		Tcl_EventuallyFree((ClientData) tracePtr, TCL_DYNAMIC); +		tracePtr = tracePtr->nextPtr; +		prevPtr->nextPtr = NULL; +		Tcl_EventuallyFree(prevPtr, TCL_DYNAMIC);  	    } -	    for (activePtr = iPtr->activeTracePtr; activePtr != NULL; -		 activePtr = activePtr->nextPtr) { -		if (activePtr->varPtr == varPtr) { +	    Tcl_DeleteHashEntry(tPtr); +	    elPtr->flags &= ~VAR_ALL_TRACES; +	    for (activePtr = iPtr->activeVarTracePtr; activePtr != NULL; +		    activePtr = activePtr->nextPtr) { +		if (activePtr->varPtr == elPtr) {  		    activePtr->nextTracePtr = NULL;  		}  	    }  	} -	     -	if (TclIsVarArray(varPtr)) { -	    DeleteArray(iPtr, Tcl_GetHashKey(tablePtr, hPtr), varPtr, -	            flags); -	    varPtr->value.tablePtr = NULL; -	} -	if (TclIsVarScalar(varPtr) && (varPtr->value.objPtr != NULL)) { -	    objPtr = varPtr->value.objPtr; -	    TclDecrRefCount(objPtr); -	    varPtr->value.objPtr = NULL; -	} -	varPtr->hPtr = NULL; -	varPtr->tracePtr = NULL; -	TclSetVarUndefined(varPtr); -	TclSetVarScalar(varPtr); - -	/* -	 * If the variable was a namespace variable, decrement its  -	 * reference count. We are in the process of destroying its -	 * namespace so that namespace will no longer "refer" to the -	 * variable. -	 */ - -	if (varPtr->flags & VAR_NAMESPACE_VAR) { -	    varPtr->flags &= ~VAR_NAMESPACE_VAR; -	    varPtr->refCount--; -	} +	TclSetVarUndefined(elPtr);  	/* -	 * Recycle the variable's memory space if there aren't any upvar's -	 * pointing to it. If there are upvars to this variable, then the -	 * variable will get freed when the last upvar goes away. +	 * Even though array elements are not supposed to be namespace +	 * variables, some combinations of [upvar] and [variable] may create +	 * such beasts - see [Bug 604239]. This is necessary to avoid leaking +	 * the corresponding Var struct, and is otherwise harmless.  	 */ -	if (varPtr->refCount == 0) { -	    ckfree((char *) varPtr); /* this Var must be VAR_IN_HASHTABLE */ -	} +	TclClearVarNamespaceVar(elPtr);      } -    Tcl_DeleteHashTable(tablePtr); +    VarHashDeleteTable(varPtr->value.tablePtr); +    ckfree(varPtr->value.tablePtr);  }  /*   *----------------------------------------------------------------------   * - * TclDeleteCompiledLocalVars -- + * TclObjVarErrMsg --   * - *	This procedure is called to recycle storage space associated with - *	the compiler-allocated array of local variables in a procedure call - *	frame. This procedure resembles TclDeleteVars above except that each - *	variable is stored in a call frame and not a hash table. For this - *	procedure to work correctly, it must not be possible for any of the - *	variable in the table to be accessed from Tcl commands (e.g. from - *	trace procedures). + *	Generate a reasonable error message describing why a variable + *	operation failed.   *   * Results:   *	None.   *   * Side effects: - *	Variables are deleted and trace procedures are invoked, if - *	any are declared. + *	The interp's result is set to hold a message identifying the variable + *	given by part1 and part2 and describing why the variable operation + *	failed.   *   *----------------------------------------------------------------------   */  void -TclDeleteCompiledLocalVars(iPtr, framePtr) -    Interp *iPtr;		/* Interpreter to which variables belong. */ -    CallFrame *framePtr;	/* Procedure call frame containing -				 * compiler-assigned local variables to -				 * delete. */ +TclVarErrMsg( +    Tcl_Interp *interp,		/* Interpreter in which to record message. */ +    const char *part1, +    const char *part2,		/* Variable's two-part name. */ +    const char *operation,	/* String describing operation that failed, +				 * e.g. "read", "set", or "unset". */ +    const char *reason)		/* String describing why operation failed. */  { -    register Var *varPtr; -    int flags;			/* Flags passed to trace procedures. */ -    Var *linkPtr; -    ActiveVarTrace *activePtr; -    int numLocals, i; +    Tcl_Obj *part2Ptr = NULL, *part1Ptr = Tcl_NewStringObj(part1, -1); -    flags = TCL_TRACE_UNSETS; -    numLocals = framePtr->numCompiledLocals; -    varPtr = framePtr->compiledLocals; -    for (i = 0;  i < numLocals;  i++) { -	/* -	 * For global/upvar variables referenced in procedures, decrement -	 * the reference count on the variable referred to, and free -	 * the referenced variable if it's no longer needed. Don't delete -	 * the hash entry for the other variable if it's in the same table -	 * as us: this will happen automatically later on. -	 */ +    if (part2) { +	part2Ptr = Tcl_NewStringObj(part2, -1); +    } -	if (TclIsVarLink(varPtr)) { -	    linkPtr = varPtr->value.linkPtr; -	    linkPtr->refCount--; -	    if ((linkPtr->refCount == 0) && TclIsVarUndefined(linkPtr) -		    && (linkPtr->tracePtr == NULL) -		    && (linkPtr->flags & VAR_IN_HASHTABLE)) { -		if (linkPtr->hPtr == NULL) { -		    ckfree((char *) linkPtr); -		} else { -		    Tcl_DeleteHashEntry(linkPtr->hPtr); -		    ckfree((char *) linkPtr); -		} -	    } +    TclObjVarErrMsg(interp, part1Ptr, part2Ptr, operation, reason, -1); + +    Tcl_DecrRefCount(part1Ptr); +    if (part2Ptr) { +	Tcl_DecrRefCount(part2Ptr); +    } +} + +void +TclObjVarErrMsg( +    Tcl_Interp *interp,		/* Interpreter in which to record message. */ +    Tcl_Obj *part1Ptr,		/* (may be NULL, if index >= 0) */ +    Tcl_Obj *part2Ptr,		/* Variable's two-part name. */ +    const char *operation,	/* String describing operation that failed, +				 * e.g. "read", "set", or "unset". */ +    const char *reason,		/* String describing why operation failed. */ +    int index)			/* Index into the local variable table of the +				 * variable, or -1. Only used when part1Ptr is +				 * NULL. */ +{ +    if (!part1Ptr) { +	if (index == -1) { +	    Tcl_Panic("invalid part1Ptr and invalid index together"); +	} +	part1Ptr = localName(((Interp *)interp)->varFramePtr, index); +    } +    Tcl_SetObjResult(interp, Tcl_ObjPrintf("can't %s \"%s%s%s%s\": %s", +	    operation, TclGetString(part1Ptr), (part2Ptr ? "(" : ""), +	    (part2Ptr ? TclGetString(part2Ptr) : ""), (part2Ptr ? ")" : ""), +	    reason)); +} + +/* + *---------------------------------------------------------------------- + * + * Internal functions for variable name object types -- + * + *---------------------------------------------------------------------- + */ + +/* + * Panic functions that should never be called in normal operation. + */ + +static void +PanicOnUpdateVarName( +    Tcl_Obj *objPtr) +{ +    Tcl_Panic("%s of type %s should not be called", "updateStringProc", +	    objPtr->typePtr->name); +} + +static int +PanicOnSetVarName( +    Tcl_Interp *interp, +    Tcl_Obj *objPtr) +{ +    Tcl_Panic("%s of type %s should not be called", "setFromAnyProc", +	    objPtr->typePtr->name); +    return TCL_ERROR; +} + +/* + * localVarName - + * + * INTERNALREP DEFINITION: + *   ptrAndLongRep.ptr:   pointer to name obj in varFramePtr->localCache + *			  or NULL if it is this same obj + *   ptrAndLongRep.value: index into locals table + */ + +static void +FreeLocalVarName( +    Tcl_Obj *objPtr) +{ +    Tcl_Obj *namePtr = objPtr->internalRep.ptrAndLongRep.ptr; + +    if (namePtr) { +	Tcl_DecrRefCount(namePtr); +    } +    objPtr->typePtr = NULL; +} + +static void +DupLocalVarName( +    Tcl_Obj *srcPtr, +    Tcl_Obj *dupPtr) +{ +    Tcl_Obj *namePtr = srcPtr->internalRep.ptrAndLongRep.ptr; + +    if (!namePtr) { +	namePtr = srcPtr; +    } +    dupPtr->internalRep.ptrAndLongRep.ptr = namePtr; +    Tcl_IncrRefCount(namePtr); + +    dupPtr->internalRep.ptrAndLongRep.value = +	    srcPtr->internalRep.ptrAndLongRep.value; +    dupPtr->typePtr = &localVarNameType; +} + +#if ENABLE_NS_VARNAME_CACHING +/* + * nsVarName - + * + * INTERNALREP DEFINITION: + *   twoPtrValue.ptr1: pointer to the namespace containing the reference. + *   twoPtrValue.ptr2: pointer to the corresponding Var + */ + +static void +FreeNsVarName( +    Tcl_Obj *objPtr) +{ +    register Var *varPtr = objPtr->internalRep.twoPtrValue.ptr2; + +    if (TclIsVarInHash(varPtr)) { +	varPtr->refCount--; +	if (TclIsVarUndefined(varPtr) && (varPtr->refCount == 0)) { +	    CleanupVar(varPtr, NULL);  	} +    } +    objPtr->typePtr = NULL; +} + +static void +DupNsVarName( +    Tcl_Obj *srcPtr, +    Tcl_Obj *dupPtr) +{ +    Namespace *nsPtr = srcPtr->internalRep.twoPtrValue.ptr1; +    register Var *varPtr = srcPtr->internalRep.twoPtrValue.ptr2; + +    dupPtr->internalRep.twoPtrValue.ptr1 = nsPtr; +    dupPtr->internalRep.twoPtrValue.ptr2 = varPtr; +    if (TclIsVarInHash(varPtr)) { +	varPtr->refCount++; +    } +    dupPtr->typePtr = &tclNsVarNameType; +} +#endif + +/* + * parsedVarName - + * + * INTERNALREP DEFINITION: + *   twoPtrValue.ptr1 = pointer to the array name Tcl_Obj (NULL if scalar) + *   twoPtrValue.ptr2 = pointer to the element name string (owned by this + *			Tcl_Obj), or NULL if it is a scalar variable + */ + +static void +FreeParsedVarName( +    Tcl_Obj *objPtr) +{ +    register Tcl_Obj *arrayPtr = objPtr->internalRep.twoPtrValue.ptr1; +    register char *elem = objPtr->internalRep.twoPtrValue.ptr2; + +    if (arrayPtr != NULL) { +	TclDecrRefCount(arrayPtr); +	ckfree(elem); +    } +    objPtr->typePtr = NULL; +} + +static void +DupParsedVarName( +    Tcl_Obj *srcPtr, +    Tcl_Obj *dupPtr) +{ +    register Tcl_Obj *arrayPtr = srcPtr->internalRep.twoPtrValue.ptr1; +    register char *elem = srcPtr->internalRep.twoPtrValue.ptr2; +    char *elemCopy; +    unsigned elemLen; + +    if (arrayPtr != NULL) { +	Tcl_IncrRefCount(arrayPtr); +	elemLen = strlen(elem); +	elemCopy = ckalloc(elemLen + 1); +	memcpy(elemCopy, elem, elemLen); +	*(elemCopy + elemLen) = '\0'; +	elem = elemCopy; +    } +    dupPtr->internalRep.twoPtrValue.ptr1 = arrayPtr; +    dupPtr->internalRep.twoPtrValue.ptr2 = elem; +    dupPtr->typePtr = &tclParsedVarNameType; +} + +static void +UpdateParsedVarName( +    Tcl_Obj *objPtr) +{ +    Tcl_Obj *arrayPtr = objPtr->internalRep.twoPtrValue.ptr1; +    char *part2 = objPtr->internalRep.twoPtrValue.ptr2; +    const char *part1; +    char *p; +    int len1, len2, totalLen; + +    if (arrayPtr == NULL) {  	/* -	 * Invoke traces on the variable that is being deleted. Then delete -	 * the variable's trace records. +	 * This is a parsed scalar name: what is it doing here?  	 */ -	if (varPtr->tracePtr != NULL) { -	    char *msg; -	    int resultType; - -	    msg = CallTraces(iPtr, (Var *) NULL, varPtr, -		    varPtr->name, (char *) NULL, flags, &resultType); -	    if (msg != NULL) { -		if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		    ckfree(msg); -		} else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    Tcl_DecrRefCount((Tcl_Obj *) msg); -		} -	    } -	    while (varPtr->tracePtr != NULL) { -		VarTrace *tracePtr = varPtr->tracePtr; -		varPtr->tracePtr = tracePtr->nextPtr; -		Tcl_EventuallyFree((ClientData) tracePtr, TCL_DYNAMIC); -	    } -	    for (activePtr = iPtr->activeTracePtr; activePtr != NULL; -		 activePtr = activePtr->nextPtr) { -		if (activePtr->varPtr == varPtr) { -		    activePtr->nextTracePtr = NULL; -		} +	Tcl_Panic("scalar parsedVarName without a string rep"); +    } + +    part1 = TclGetStringFromObj(arrayPtr, &len1); +    len2 = strlen(part2); + +    totalLen = len1 + len2 + 2; +    p = ckalloc(totalLen + 1); +    objPtr->bytes = p; +    objPtr->length = totalLen; + +    memcpy(p, part1, (unsigned) len1); +    p += len1; +    *p++ = '('; +    memcpy(p, part2, (unsigned) len2); +    p += len2; +    *p++ = ')'; +    *p = '\0'; +} + +/* + *---------------------------------------------------------------------- + * + * Tcl_FindNamespaceVar -- MOVED OVER from tclNamesp.c + * + *	Searches for a namespace variable, a variable not local to a + *	procedure. The variable can be either a scalar or an array, but may + *	not be an element of an array. + * + * Results: + *	Returns a token for the variable if it is found. Otherwise, if it + *	can't be found or there is an error, returns NULL and leaves an error + *	message in the interpreter's result object if "flags" contains + *	TCL_LEAVE_ERR_MSG. + * + * Side effects: + *	None. + * + *---------------------------------------------------------------------- + */ + +Tcl_Var +Tcl_FindNamespaceVar( +    Tcl_Interp *interp,		/* The interpreter in which to find the +				 * variable. */ +    const char *name,		/* Variable's name. If it starts with "::", +				 * will be looked up in global namespace. +				 * Else, looked up first in contextNsPtr +				 * (current namespace if contextNsPtr is +				 * NULL), then in global namespace. */ +    Tcl_Namespace *contextNsPtr,/* Ignored if TCL_GLOBAL_ONLY flag set. +				 * Otherwise, points to namespace in which to +				 * resolve name. If NULL, look up name in the +				 * current namespace. */ +    int flags)			/* An OR'd combination of: AVOID_RESOLVERS, +				 * TCL_GLOBAL_ONLY (look up name only in +				 * global namespace), TCL_NAMESPACE_ONLY (look +				 * up only in contextNsPtr, or the current +				 * namespace if contextNsPtr is NULL), and +				 * TCL_LEAVE_ERR_MSG. If both TCL_GLOBAL_ONLY +				 * and TCL_NAMESPACE_ONLY are given, +				 * TCL_GLOBAL_ONLY is ignored. */ +{ +    Tcl_Obj *namePtr = Tcl_NewStringObj(name, -1); +    Tcl_Var var; + +    var = ObjFindNamespaceVar(interp, namePtr, contextNsPtr, flags); +    Tcl_DecrRefCount(namePtr); +    return var; +} + +static Tcl_Var +ObjFindNamespaceVar( +    Tcl_Interp *interp,		/* The interpreter in which to find the +				 * variable. */ +    Tcl_Obj *namePtr,		/* Variable's name. If it starts with "::", +				 * will be looked up in global namespace. +				 * Else, looked up first in contextNsPtr +				 * (current namespace if contextNsPtr is +				 * NULL), then in global namespace. */ +    Tcl_Namespace *contextNsPtr,/* Ignored if TCL_GLOBAL_ONLY flag set. +				 * Otherwise, points to namespace in which to +				 * resolve name. If NULL, look up name in the +				 * current namespace. */ +    int flags)			/* An OR'd combination of: AVOID_RESOLVERS, +				 * TCL_GLOBAL_ONLY (look up name only in +				 * global namespace), TCL_NAMESPACE_ONLY (look +				 * up only in contextNsPtr, or the current +				 * namespace if contextNsPtr is NULL), and +				 * TCL_LEAVE_ERR_MSG. If both TCL_GLOBAL_ONLY +				 * and TCL_NAMESPACE_ONLY are given, +				 * TCL_GLOBAL_ONLY is ignored. */ +{ +    Interp *iPtr = (Interp *) interp; +    ResolverScheme *resPtr; +    Namespace *nsPtr[2], *cxtNsPtr; +    const char *simpleName; +    Var *varPtr; +    register int search; +    int result; +    Tcl_Var var; +    Tcl_Obj *simpleNamePtr; +    const char *name = TclGetString(namePtr); + +    /* +     * If this namespace has a variable resolver, then give it first crack at +     * the variable resolution. It may return a Tcl_Var value, it may signal +     * to continue onward, or it may signal an error. +     */ + +    if ((flags & TCL_GLOBAL_ONLY) != 0) { +	cxtNsPtr = (Namespace *) TclGetGlobalNamespace(interp); +    } else if (contextNsPtr != NULL) { +	cxtNsPtr = (Namespace *) contextNsPtr; +    } else { +	cxtNsPtr = (Namespace *) TclGetCurrentNamespace(interp); +    } + +    if (!(flags & AVOID_RESOLVERS) && +	    (cxtNsPtr->varResProc != NULL || iPtr->resolverPtr != NULL)) { +	resPtr = iPtr->resolverPtr; + +	if (cxtNsPtr->varResProc) { +	    result = cxtNsPtr->varResProc(interp, name, +		    (Tcl_Namespace *) cxtNsPtr, flags, &var); +	} else { +	    result = TCL_CONTINUE; +	} + +	while (result == TCL_CONTINUE && resPtr) { +	    if (resPtr->varResProc) { +		result = resPtr->varResProc(interp, name, +			(Tcl_Namespace *) cxtNsPtr, flags, &var);  	    } +	    resPtr = resPtr->nextPtr;  	} -        /* -	 * Now if the variable is an array, delete its element hash table. -	 * Otherwise, if it's a scalar variable, decrement the ref count -	 * of its value. -	 */ -	     -	if (TclIsVarArray(varPtr) && (varPtr->value.tablePtr != NULL)) { -	    DeleteArray(iPtr, varPtr->name, varPtr, flags); +	if (result == TCL_OK) { +	    return var; +	} else if (result != TCL_CONTINUE) { +	    return NULL;  	} -	if (TclIsVarScalar(varPtr) && (varPtr->value.objPtr != NULL)) { -	    TclDecrRefCount(varPtr->value.objPtr); -	    varPtr->value.objPtr = NULL; +    } + +    /* +     * Find the namespace(s) that contain the variable. +     */ + +    TclGetNamespaceForQualName(interp, name, (Namespace *) contextNsPtr, +	    flags, &nsPtr[0], &nsPtr[1], &cxtNsPtr, &simpleName); + +    /* +     * Look for the variable in the variable table of its namespace. Be sure +     * to check both possible search paths: from the specified namespace +     * context and from the global namespace. +     */ + +    varPtr = NULL; +    if (simpleName != name) { +	simpleNamePtr = Tcl_NewStringObj(simpleName, -1); +    } else { +	simpleNamePtr = namePtr; +    } + +    for (search = 0;  (search < 2) && (varPtr == NULL);  search++) { +	if ((nsPtr[search] != NULL) && (simpleName != NULL)) { +	    varPtr = VarHashFindVar(&nsPtr[search]->varTable, simpleNamePtr);  	} -	varPtr->hPtr = NULL; -	varPtr->tracePtr = NULL; -	TclSetVarUndefined(varPtr); -	TclSetVarScalar(varPtr); -	varPtr++;      } +    if (simpleName != name) { +	Tcl_DecrRefCount(simpleNamePtr); +    } +    if ((varPtr == NULL) && (flags & TCL_LEAVE_ERR_MSG)) { +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"unknown variable \"%s\"", name)); +	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "VARIABLE", name, NULL); +    } +    return (Tcl_Var) varPtr;  }  /*   *----------------------------------------------------------------------   * - * DeleteArray -- + * InfoVarsCmd -- (moved over from tclCmdIL.c) + * + *	Called to implement the "info vars" command that returns the list of + *	variables in the interpreter that match an optional pattern. The + *	pattern, if any, consists of an optional sequence of namespace names + *	separated by "::" qualifiers, which is followed by a glob-style + *	pattern that restricts which variables are returned. Handles the + *	following syntax:   * - *	This procedure is called to free up everything in an array - *	variable.  It's the caller's responsibility to make sure - *	that the array is no longer accessible before this procedure - *	is called. + *	    info vars ?pattern?   *   * Results: - *	None. + *	Returns TCL_OK if successful and TCL_ERROR if there is an error.   *   * Side effects: - *	All storage associated with varPtr's array elements is deleted - *	(including the array's hash table). Deletion trace procedures for - *	array elements are invoked, then deleted. Any pending traces for - *	array elements are also deleted. + *	Returns a result in the interpreter's result object. If there is an + *	error, the result is an error message.   *   *----------------------------------------------------------------------   */ -static void -DeleteArray(iPtr, arrayName, varPtr, flags) -    Interp *iPtr;			/* Interpreter containing array. */ -    char *arrayName;			/* Name of array (used for trace -					 * callbacks). */ -    Var *varPtr;			/* Pointer to variable structure. */ -    int flags;				/* Flags to pass to CallTraces: -					 * TCL_TRACE_UNSETS and sometimes -					 * TCL_INTERP_DESTROYED, -					 * TCL_NAMESPACE_ONLY, or -					 * TCL_GLOBAL_ONLY. */ +int +TclInfoVarsCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  { +    Interp *iPtr = (Interp *) interp; +    const char *varName, *pattern, *simplePattern;      Tcl_HashSearch search; -    register Tcl_HashEntry *hPtr; -    register Var *elPtr; -    ActiveVarTrace *activePtr; -    Tcl_Obj *objPtr; +    Var *varPtr; +    Namespace *nsPtr; +    Namespace *globalNsPtr = (Namespace *) Tcl_GetGlobalNamespace(interp); +    Namespace *currNsPtr = (Namespace *) Tcl_GetCurrentNamespace(interp); +    Tcl_Obj *listPtr, *elemObjPtr, *varNamePtr; +    int specificNsInPattern = 0;/* Init. to avoid compiler warning. */ +    Tcl_Obj *simplePatternPtr = NULL; -    DeleteSearches(varPtr); -    for (hPtr = Tcl_FirstHashEntry(varPtr->value.tablePtr, &search); -	 hPtr != NULL;  hPtr = Tcl_NextHashEntry(&search)) { -	elPtr = (Var *) Tcl_GetHashValue(hPtr); -	if (TclIsVarScalar(elPtr) && (elPtr->value.objPtr != NULL)) { -	    objPtr = elPtr->value.objPtr; -	    TclDecrRefCount(objPtr); -	    elPtr->value.objPtr = NULL; +    /* +     * Get the pattern and find the "effective namespace" in which to list +     * variables. We only use this effective namespace if there's no active +     * Tcl procedure frame. +     */ + +    if (objc == 1) { +	simplePattern = NULL; +	nsPtr = currNsPtr; +	specificNsInPattern = 0; +    } else if (objc == 2) { +	/* +	 * From the pattern, get the effective namespace and the simple +	 * pattern (no namespace qualifiers or ::'s) at the end. If an error +	 * was found while parsing the pattern, return it. Otherwise, if the +	 * namespace wasn't found, just leave nsPtr NULL: we will return an +	 * empty list since no variables there can be found. +	 */ + +	Namespace *dummy1NsPtr, *dummy2NsPtr; + +	pattern = TclGetString(objv[1]); +	TclGetNamespaceForQualName(interp, pattern, NULL, /*flags*/ 0, +		&nsPtr, &dummy1NsPtr, &dummy2NsPtr, &simplePattern); + +	if (nsPtr != NULL) {	/* We successfully found the pattern's ns. */ +	    specificNsInPattern = (strcmp(simplePattern, pattern) != 0); +	    if (simplePattern == pattern) { +		simplePatternPtr = objv[1]; +	    } else { +		simplePatternPtr = Tcl_NewStringObj(simplePattern, -1); +	    } +	    Tcl_IncrRefCount(simplePatternPtr);  	} -	elPtr->hPtr = NULL; -	if (elPtr->tracePtr != NULL) { -	    char *msg; -	    int resultType; - -	    elPtr->flags &= ~VAR_TRACE_ACTIVE; -	    msg = CallTraces(iPtr, (Var *) NULL, elPtr, arrayName, -		    Tcl_GetHashKey(varPtr->value.tablePtr, hPtr), flags, -		    &resultType); -	    if (msg != NULL) { -		if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		    ckfree(msg); -		} else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		    Tcl_DecrRefCount((Tcl_Obj *) msg); +    } else { +	Tcl_WrongNumArgs(interp, 1, objv, "?pattern?"); +	return TCL_ERROR; +    } + +    /* +     * If the namespace specified in the pattern wasn't found, just return. +     */ + +    if (nsPtr == NULL) { +	return TCL_OK; +    } + +    listPtr = Tcl_NewListObj(0, NULL); + +    if (!HasLocalVars(iPtr->varFramePtr) || specificNsInPattern) { +	/* +	 * There is no frame pointer, the frame pointer was pushed only to +	 * activate a namespace, or we are in a procedure call frame but a +	 * specific namespace was specified. Create a list containing only the +	 * variables in the effective namespace's variable table. +	 */ + +	if (simplePattern && TclMatchIsTrivial(simplePattern)) { +	    /* +	     * If we can just do hash lookups, that simplifies things a lot. +	     */ + +	    varPtr = VarHashFindVar(&nsPtr->varTable, simplePatternPtr); +	    if (varPtr) { +		if (!TclIsVarUndefined(varPtr) +			|| TclIsVarNamespaceVar(varPtr)) { +		    if (specificNsInPattern) { +			elemObjPtr = Tcl_NewObj(); +			Tcl_GetVariableFullName(interp, (Tcl_Var) varPtr, +				elemObjPtr); +		    } else { +			elemObjPtr = VarHashGetKey(varPtr); +		    } +		    Tcl_ListObjAppendElement(interp, listPtr, elemObjPtr); +		} +	    } else if ((nsPtr != globalNsPtr) && !specificNsInPattern) { +		varPtr = VarHashFindVar(&globalNsPtr->varTable, +			simplePatternPtr); +		if (varPtr) { +		    if (!TclIsVarUndefined(varPtr) +			    || TclIsVarNamespaceVar(varPtr)) { +			Tcl_ListObjAppendElement(interp, listPtr, +				VarHashGetKey(varPtr)); +		    }  		}  	    } -	    while (elPtr->tracePtr != NULL) { -		VarTrace *tracePtr = elPtr->tracePtr; -		elPtr->tracePtr = tracePtr->nextPtr; -		Tcl_EventuallyFree((ClientData) tracePtr,TCL_DYNAMIC); +	} else { +	    /* +	     * Have to scan the tables of variables. +	     */ + +	    varPtr = VarHashFirstVar(&nsPtr->varTable, &search); +	    while (varPtr) { +		if (!TclIsVarUndefined(varPtr) +			|| TclIsVarNamespaceVar(varPtr)) { +		    varNamePtr = VarHashGetKey(varPtr); +		    varName = TclGetString(varNamePtr); +		    if ((simplePattern == NULL) +			    || Tcl_StringMatch(varName, simplePattern)) { +			if (specificNsInPattern) { +			    elemObjPtr = Tcl_NewObj(); +			    Tcl_GetVariableFullName(interp, (Tcl_Var) varPtr, +				    elemObjPtr); +			} else { +			    elemObjPtr = varNamePtr; +			} +			Tcl_ListObjAppendElement(interp, listPtr, elemObjPtr); +		    } +		} +		varPtr = VarHashNextVar(&search);  	    } -	    for (activePtr = iPtr->activeTracePtr; activePtr != NULL; -		 activePtr = activePtr->nextPtr) { -		if (activePtr->varPtr == elPtr) { -		    activePtr->nextTracePtr = NULL; + +	    /* +	     * If the effective namespace isn't the global :: namespace, and a +	     * specific namespace wasn't requested in the pattern (i.e., the +	     * pattern only specifies variable names), then add in all global +	     * :: variables that match the simple pattern. Of course, add in +	     * only those variables that aren't hidden by a variable in the +	     * effective namespace. +	     */ + +	    if ((nsPtr != globalNsPtr) && !specificNsInPattern) { +		varPtr = VarHashFirstVar(&globalNsPtr->varTable,&search); +		while (varPtr) { +		    if (!TclIsVarUndefined(varPtr) +			    || TclIsVarNamespaceVar(varPtr)) { +			varNamePtr = VarHashGetKey(varPtr); +			varName = TclGetString(varNamePtr); +			if ((simplePattern == NULL) +				|| Tcl_StringMatch(varName, simplePattern)) { +			    if (VarHashFindVar(&nsPtr->varTable, +				    varNamePtr) == NULL) { +				Tcl_ListObjAppendElement(interp, listPtr, +					varNamePtr); +			    } +			} +		    } +		    varPtr = VarHashNextVar(&search);  		}  	    }  	} -	TclSetVarUndefined(elPtr); -	TclSetVarScalar(elPtr); -	if (elPtr->refCount == 0) { -	    ckfree((char *) elPtr); /* element Vars are VAR_IN_HASHTABLE */ -	} +    } else if (iPtr->varFramePtr->procPtr != NULL) { +	AppendLocals(interp, listPtr, simplePatternPtr, 1); +    } + +    if (simplePatternPtr) { +	Tcl_DecrRefCount(simplePatternPtr);      } -    Tcl_DeleteHashTable(varPtr->value.tablePtr); -    ckfree((char *) varPtr->value.tablePtr); +    Tcl_SetObjResult(interp, listPtr); +    return TCL_OK;  }  /*   *----------------------------------------------------------------------   * - * CleanupVar -- + * InfoGlobalsCmd -- (moved over from tclCmdIL.c)   * - *	This procedure is called when it looks like it may be OK to free up - *	a variable's storage. If the variable is in a hashtable, its Var - *	structure and hash table entry will be freed along with those of its - *	containing array, if any. This procedure is called, for example, - *	when a trace on a variable deletes a variable. + *	Called to implement the "info globals" command that returns the list + *	of global variables matching an optional pattern. Handles the + *	following syntax: + * + *	    info globals ?pattern?   *   * Results: - *	None. + *	Returns TCL_OK if successful and TCL_ERROR if there is an error.   *   * Side effects: - *	If the variable (or its containing array) really is dead and in a - *	hashtable, then its Var structure, and possibly its hash table - *	entry, is freed up. + *	Returns a result in the interpreter's result object. If there is an + *	error, the result is an error message.   *   *----------------------------------------------------------------------   */ -static void -CleanupVar(varPtr, arrayPtr) -    Var *varPtr;		/* Pointer to variable that may be a -				 * candidate for being expunged. */ -    Var *arrayPtr;		/* Array that contains the variable, or -				 * NULL if this variable isn't an array -				 * element. */ +int +TclInfoGlobalsCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  { -    if (TclIsVarUndefined(varPtr) && (varPtr->refCount == 0) -	    && (varPtr->tracePtr == NULL) -	    && (varPtr->flags & VAR_IN_HASHTABLE)) { -	if (varPtr->hPtr != NULL) { -	    Tcl_DeleteHashEntry(varPtr->hPtr); +    const char *varName, *pattern; +    Namespace *globalNsPtr = (Namespace *) Tcl_GetGlobalNamespace(interp); +    Tcl_HashSearch search; +    Var *varPtr; +    Tcl_Obj *listPtr, *varNamePtr, *patternPtr; + +    if (objc == 1) { +	pattern = NULL; +    } else if (objc == 2) { +	pattern = TclGetString(objv[1]); + +	/* +	 * Strip leading global-namespace qualifiers. [Bug 1057461] +	 */ + +	if (pattern[0] == ':' && pattern[1] == ':') { +	    while (*pattern == ':') { +		pattern++; +	    }  	} -	ckfree((char *) varPtr); +    } else { +	Tcl_WrongNumArgs(interp, 1, objv, "?pattern?"); +	return TCL_ERROR;      } -    if (arrayPtr != NULL) { -	if (TclIsVarUndefined(arrayPtr) && (arrayPtr->refCount == 0) -		&& (arrayPtr->tracePtr == NULL) -	        && (arrayPtr->flags & VAR_IN_HASHTABLE)) { -	    if (arrayPtr->hPtr != NULL) { -		Tcl_DeleteHashEntry(arrayPtr->hPtr); + +    /* +     * Scan through the global :: namespace's variable table and create a list +     * of all global variables that match the pattern. +     */ + +    listPtr = Tcl_NewListObj(0, NULL); +    if (pattern != NULL && TclMatchIsTrivial(pattern)) { +	if (pattern == TclGetString(objv[1])) { +	    patternPtr = objv[1]; +	} else { +	    patternPtr = Tcl_NewStringObj(pattern, -1); +	} +	Tcl_IncrRefCount(patternPtr); + +	varPtr = VarHashFindVar(&globalNsPtr->varTable, patternPtr); +	if (varPtr) { +	    if (!TclIsVarUndefined(varPtr)) { +		Tcl_ListObjAppendElement(interp, listPtr, +			VarHashGetKey(varPtr)); +	    } +	} +	Tcl_DecrRefCount(patternPtr); +    } else { +	for (varPtr = VarHashFirstVar(&globalNsPtr->varTable, &search); +		varPtr != NULL; +		varPtr = VarHashNextVar(&search)) { +	    if (TclIsVarUndefined(varPtr)) { +		continue; +	    } +	    varNamePtr = VarHashGetKey(varPtr); +	    varName = TclGetString(varNamePtr); +	    if ((pattern == NULL) || Tcl_StringMatch(varName, pattern)) { +		Tcl_ListObjAppendElement(interp, listPtr, varNamePtr);  	    } -	    ckfree((char *) arrayPtr);  	}      } +    Tcl_SetObjResult(interp, listPtr); +    return TCL_OK;  } +  /*   *----------------------------------------------------------------------   * - * VarErrMsg -- + * TclInfoLocalsCmd -- (moved over from tclCmdIl.c) + * + *	Called to implement the "info locals" command to return a list of + *	local variables that match an optional pattern. Handles the following + *	syntax:   * - *      Generate a reasonable error message describing why a variable - *      operation failed. + *	    info locals ?pattern?   *   * Results: - *      None. + *	Returns TCL_OK if successful and TCL_ERROR if there is an error.   *   * Side effects: - *      The interp's result is set to hold a message identifying the - *      variable given by part1 and part2 and describing why the - *      variable operation failed. + *	Returns a result in the interpreter's result object. If there is an + *	error, the result is an error message.   *   *----------------------------------------------------------------------   */ -static void -VarErrMsg(interp, part1, part2, operation, reason) -    Tcl_Interp *interp;         /* Interpreter in which to record message. */ -    char *part1, *part2;        /* Variable's two-part name. */ -    char *operation;            /* String describing operation that failed, -                                 * e.g. "read", "set", or "unset". */ -    char *reason;               /* String describing why operation failed. */ +int +TclInfoLocalsCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Argument objects. */  { -    Tcl_ResetResult(interp); -    Tcl_AppendResult(interp, "can't ", operation, " \"", part1, -	    (char *) NULL); -    if (part2 != NULL) { -        Tcl_AppendResult(interp, "(", part2, ")", (char *) NULL); +    Interp *iPtr = (Interp *) interp; +    Tcl_Obj *patternPtr, *listPtr; + +    if (objc == 1) { +	patternPtr = NULL; +    } else if (objc == 2) { +	patternPtr = objv[1]; +    } else { +	Tcl_WrongNumArgs(interp, 1, objv, "?pattern?"); +	return TCL_ERROR;      } -    Tcl_AppendResult(interp, "\": ", reason, (char *) NULL); -} +    if (!HasLocalVars(iPtr->varFramePtr)) { +	return TCL_OK; +    } + +    /* +     * Return a list containing names of first the compiled locals (i.e. the +     * ones stored in the call frame), then the variables in the local hash +     * table (if one exists). +     */ + +    listPtr = Tcl_NewListObj(0, NULL); +    AppendLocals(interp, listPtr, patternPtr, 0); +    Tcl_SetObjResult(interp, listPtr); +    return TCL_OK; +}  /*   *----------------------------------------------------------------------   * - * TclTraceVarExists -- + * AppendLocals --   * - *	This is called from info exists.  We need to trigger read - *	and/or array traces because they may end up creating a - *	variable that doesn't currently exist. + *	Append the local variables for the current frame to the specified list + *	object.   *   * Results: - *	A pointer to the Var structure, or NULL. + *	None.   *   * Side effects: - *	May fill in error messages in the interp. + *	None.   *   *----------------------------------------------------------------------   */ -Var * -TclVarTraceExists(interp, varName) -    Tcl_Interp *interp;		/* The interpreter */ -    char *varName;		/* The variable name */ +static void +AppendLocals( +    Tcl_Interp *interp,		/* Current interpreter. */ +    Tcl_Obj *listPtr,		/* List object to append names to. */ +    Tcl_Obj *patternPtr,	/* Pattern to match against. */ +    int includeLinks)		/* 1 if upvars should be included, else 0. */  { +    Interp *iPtr = (Interp *) interp;      Var *varPtr; -    Var *arrayPtr; +    int i, localVarCt, added; +    Tcl_Obj **varNamePtr, *objNamePtr; +    const char *varName; +    TclVarHashTable *localVarTablePtr; +    Tcl_HashSearch search; +    Tcl_HashTable addedTable; +    const char *pattern = patternPtr? TclGetString(patternPtr) : NULL; + +    localVarCt = iPtr->varFramePtr->numCompiledLocals; +    varPtr = iPtr->varFramePtr->compiledLocals; +    localVarTablePtr = iPtr->varFramePtr->varTablePtr; +    varNamePtr = &iPtr->varFramePtr->localCachePtr->varName0; +    if (includeLinks) { +	Tcl_InitObjHashTable(&addedTable); +    } + +    for (i = 0; i < localVarCt; i++, varNamePtr++) { +	/* +	 * Skip nameless (temporary) variables and undefined variables. +	 */ + +	if (*varNamePtr && !TclIsVarUndefined(varPtr) +		&& (includeLinks || !TclIsVarLink(varPtr))) { +	    varName = TclGetString(*varNamePtr); +	    if ((pattern == NULL) || Tcl_StringMatch(varName, pattern)) { +		Tcl_ListObjAppendElement(interp, listPtr, *varNamePtr); +		if (includeLinks) { +		    Tcl_CreateHashEntry(&addedTable, *varNamePtr, &added); +		} +	    } +	} +	varPtr++; +    }      /* -     * The choice of "create" flag values is delicate here, and -     * matches the semantics of GetVar.  Things are still not perfect, -     * however, because if you do "info exists x" you get a varPtr -     * and therefore trigger traces.  However, if you do  -     * "info exists x(i)", then you only get a varPtr if x is already -     * known to be an array.  Otherwise you get NULL, and no trace -     * is triggered.  This matches Tcl 7.6 semantics. +     * Do nothing if no local variables.       */ -    varPtr = TclLookupVar(interp, varName, (char *) NULL, -            0, "access", /*createPart1*/ 0, /*createPart2*/ 1, &arrayPtr); +    if (localVarTablePtr == NULL) { +	goto objectVars; +    } + +    /* +     * Check for the simple and fast case. +     */ -    if (varPtr == NULL) { -	return NULL; +    if ((pattern != NULL) && TclMatchIsTrivial(pattern)) { +	varPtr = VarHashFindVar(localVarTablePtr, patternPtr); +	if (varPtr != NULL) { +	    if (!TclIsVarUndefined(varPtr) +		    && (includeLinks || !TclIsVarLink(varPtr))) { +		Tcl_ListObjAppendElement(interp, listPtr, +			VarHashGetKey(varPtr)); +		if (includeLinks) { +		    Tcl_CreateHashEntry(&addedTable, VarHashGetKey(varPtr), +			    &added); +		} +	    } +	} +	goto objectVars;      } -    if ((varPtr->tracePtr != NULL) -	    || ((arrayPtr != NULL) && (arrayPtr->tracePtr != NULL))) { -	char *msg; -	int resultType; +    /* +     * Scan over and process all local variables. +     */ -	msg = CallTraces((Interp *)interp, arrayPtr, varPtr, varName, -		(char *) NULL, TCL_TRACE_READS, &resultType); -	if (msg != NULL) { -	    if (resultType & TCL_TRACE_RESULT_DYNAMIC) { -		ckfree(msg); -	    } else if (resultType & TCL_TRACE_RESULT_OBJECT) { -		Tcl_DecrRefCount((Tcl_Obj *) msg); +    for (varPtr = VarHashFirstVar(localVarTablePtr, &search); +	    varPtr != NULL; +	    varPtr = VarHashNextVar(&search)) { +	if (!TclIsVarUndefined(varPtr) +		&& (includeLinks || !TclIsVarLink(varPtr))) { +	    objNamePtr = VarHashGetKey(varPtr); +	    varName = TclGetString(objNamePtr); +	    if ((pattern == NULL) || Tcl_StringMatch(varName, pattern)) { +		Tcl_ListObjAppendElement(interp, listPtr, objNamePtr); +		if (includeLinks) { +		    Tcl_CreateHashEntry(&addedTable, objNamePtr, &added); +		}  	    }  	}      } +  objectVars: +    if (!includeLinks) { +	return; +    } + +    if (iPtr->varFramePtr->isProcCallFrame & FRAME_IS_METHOD) { +	CallContext *contextPtr = iPtr->varFramePtr->clientData; +	Method *mPtr = contextPtr->callPtr->chain[contextPtr->index].mPtr; + +	if (mPtr->declaringObjectPtr) { +	    FOREACH(objNamePtr, mPtr->declaringObjectPtr->variables) { +		Tcl_CreateHashEntry(&addedTable, objNamePtr, &added); +		if (added && (!pattern || +			Tcl_StringMatch(TclGetString(objNamePtr), pattern))) { +		    Tcl_ListObjAppendElement(interp, listPtr, objNamePtr); +		} +	    } +	} else { +	    FOREACH(objNamePtr, mPtr->declaringClassPtr->variables) { +		Tcl_CreateHashEntry(&addedTable, objNamePtr, &added); +		if (added && (!pattern || +			Tcl_StringMatch(TclGetString(objNamePtr), pattern))) { +		    Tcl_ListObjAppendElement(interp, listPtr, objNamePtr); +		} +	    } +	} +    } +    Tcl_DeleteHashTable(&addedTable); +} + +/* + * Hash table implementation - first, just copy and adapt the obj key stuff + */ + +void +TclInitVarHashTable( +    TclVarHashTable *tablePtr, +    Namespace *nsPtr) +{ +    Tcl_InitCustomHashTable(&tablePtr->table, +	    TCL_CUSTOM_TYPE_KEYS, &tclVarHashKeyType); +    tablePtr->nsPtr = nsPtr; +} + +static Tcl_HashEntry * +AllocVarEntry( +    Tcl_HashTable *tablePtr,	/* Hash table. */ +    void *keyPtr)		/* Key to store in the hash table entry. */ +{ +    Tcl_Obj *objPtr = keyPtr; +    Tcl_HashEntry *hPtr; +    Var *varPtr; + +    varPtr = ckalloc(sizeof(VarInHash)); +    varPtr->flags = VAR_IN_HASHTABLE; +    varPtr->value.objPtr = NULL; +    VarHashRefCount(varPtr) = 1; + +    hPtr = &(((VarInHash *) varPtr)->entry); +    Tcl_SetHashValue(hPtr, varPtr); +    hPtr->key.objPtr = objPtr; +    Tcl_IncrRefCount(objPtr); + +    return hPtr; +} + +static void +FreeVarEntry( +    Tcl_HashEntry *hPtr) +{ +    Var *varPtr = VarHashGetValue(hPtr); +    Tcl_Obj *objPtr = hPtr->key.objPtr; + +    if (TclIsVarUndefined(varPtr) && !TclIsVarTraced(varPtr) +	    && (VarHashRefCount(varPtr) == 1)) { +	ckfree(varPtr); +    } else { +	VarHashInvalidateEntry(varPtr); +	TclSetVarUndefined(varPtr); +	VarHashRefCount(varPtr)--; +    } +    Tcl_DecrRefCount(objPtr); +} + +static int +CompareVarKeys( +    void *keyPtr,		/* New key to compare. */ +    Tcl_HashEntry *hPtr)	/* Existing key to compare. */ +{ +    Tcl_Obj *objPtr1 = keyPtr; +    Tcl_Obj *objPtr2 = hPtr->key.objPtr; +    register const char *p1, *p2; +    register int l1, l2; +      /* -     * If the variable doesn't exist anymore and no-one's using -     * it, then free up the relevant structures and hash table entries. +     * If the object pointers are the same then they match.       */ -    if (TclIsVarUndefined(varPtr)) { -	CleanupVar(varPtr, arrayPtr); -	return NULL; +    if (objPtr1 == objPtr2) { +	return 1;      } -    return varPtr; +    /* +     * Don't use Tcl_GetStringFromObj as it would prevent l1 and l2 being in a +     * register. +     */ + +    p1 = TclGetString(objPtr1); +    l1 = objPtr1->length; +    p2 = TclGetString(objPtr2); +    l2 = objPtr2->length; + +    /* +     * Only compare string representations of the same length. +     */ + +    return ((l1 == l2) && !memcmp(p1, p2, l1));  } + +/* + * Local Variables: + * mode: c + * c-basic-offset: 4 + * fill-column: 78 + * End: + */ | 
