diff options
Diffstat (limited to 'generic/tclEnsemble.c')
| -rw-r--r-- | generic/tclEnsemble.c | 881 | 
1 files changed, 698 insertions, 183 deletions
| diff --git a/generic/tclEnsemble.c b/generic/tclEnsemble.c index bc9ff16..9bb7a0c 100644 --- a/generic/tclEnsemble.c +++ b/generic/tclEnsemble.c @@ -4,7 +4,7 @@   *	Contains support for ensembles (see TIP#112), which provide simple   *	mechanism for creating composite commands on top of namespaces.   * - * Copyright (c) 2005-2010 Donal K. Fellows. + * Copyright (c) 2005-2013 Donal K. Fellows.   *   * See the file "license.terms" for information on usage and redistribution of   * this file, and for a DISCLAIMER OF ALL WARRANTIES. @@ -17,6 +17,7 @@   * Declarations for functions local to this file:   */ +static inline Tcl_Obj *	NewNsObj(Tcl_Namespace *namespacePtr);  static inline int	EnsembleUnknownCallback(Tcl_Interp *interp,  			    EnsembleConfig *ensemblePtr, int objc,  			    Tcl_Obj *const objv[], Tcl_Obj **prefixObjPtr); @@ -34,6 +35,12 @@ static void		MakeCachedEnsembleCommand(Tcl_Obj *objPtr,  static void		FreeEnsembleCmdRep(Tcl_Obj *objPtr);  static void		DupEnsembleCmdRep(Tcl_Obj *objPtr, Tcl_Obj *copyPtr);  static void		StringOfEnsembleCmdRep(Tcl_Obj *objPtr); +static void		CompileToInvokedCommand(Tcl_Interp *interp, +			    Tcl_Parse *parsePtr, Tcl_Obj *replacements, +			    Command *cmdPtr, CompileEnv *envPtr); +static int		CompileBasicNArgCommand(Tcl_Interp *interp, +			    Tcl_Parse *parsePtr, Command *cmdPtr, +			    CompileEnv *envPtr);  /*   * The lists of subcommands and options for the [namespace ensemble] command. @@ -77,6 +84,20 @@ const Tcl_ObjType tclEnsembleCmdType = {      StringOfEnsembleCmdRep,	/* updateStringProc */      NULL			/* setFromAnyProc */  }; + + +static inline Tcl_Obj * +NewNsObj( +    Tcl_Namespace *namespacePtr) +{ +    register Namespace *nsPtr = (Namespace *) namespacePtr; + +    if (namespacePtr == TclGetGlobalNamespace(nsPtr->interp)) { +	return Tcl_NewStringObj("::", 2); +    } else { +	return Tcl_NewStringObj(nsPtr->fullName, -1); +    } +}  /*   *---------------------------------------------------------------------- @@ -116,18 +137,19 @@ TclNamespaceEnsembleCmd(      if (nsPtr == NULL || nsPtr->flags & NS_DYING) {  	if (!Tcl_InterpDeleted(interp)) { -	    Tcl_AppendResult(interp, +	    Tcl_SetObjResult(interp, Tcl_NewStringObj(  		    "tried to manipulate ensemble of deleted namespace", -		    NULL); +		    -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "DEAD", NULL);  	}  	return TCL_ERROR;      } -    if (objc < 3) { -	Tcl_WrongNumArgs(interp, 2, objv, "subcommand ?arg ...?"); +    if (objc < 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "subcommand ?arg ...?");  	return TCL_ERROR;      } -    if (Tcl_GetIndexFromObj(interp, objv[2], ensembleSubcommands, +    if (Tcl_GetIndexFromObj(interp, objv[1], ensembleSubcommands,  	    "subcommand", 0, &index) != TCL_OK) {  	return TCL_ERROR;      } @@ -149,12 +171,12 @@ TclNamespaceEnsembleCmd(  	 * Check that we've got option-value pairs... [Bug 1558654]  	 */ -	if ((objc & 1) == 0) { -	    Tcl_WrongNumArgs(interp, 3, objv, "?option value ...?"); +	if (objc & 1) { +	    Tcl_WrongNumArgs(interp, 2, objv, "?option value ...?");  	    return TCL_ERROR;  	} -	objv += 3; -	objc -= 3; +	objv += 2; +	objc -= 2;  	/*  	 * Work out what name to use for the command to create. If supplied, @@ -235,9 +257,11 @@ TclNamespaceEnsembleCmd(  			return TCL_ERROR;  		    }  		    if (len < 1) { -			Tcl_SetResult(interp, +			Tcl_SetObjResult(interp, Tcl_NewStringObj(  				"ensemble subcommand implementations " -				"must be non-empty lists", TCL_STATIC); +				"must be non-empty lists", -1)); +			Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", +				"EMPTY_TARGET", NULL);  			Tcl_DictObjDone(&search);  			if (patchedDict) {  			    Tcl_DecrRefCount(patchedDict); @@ -250,7 +274,7 @@ TclNamespaceEnsembleCmd(  		    cmd = TclGetString(listv[0]);  		    if (!(cmd[0] == ':' && cmd[1] == ':')) {  			Tcl_Obj *newList = Tcl_NewListObj(len, listv); -			Tcl_Obj *newCmd = Tcl_NewStringObj(nsPtr->fullName,-1); +			Tcl_Obj *newCmd = NewNsObj((Tcl_Namespace *) nsPtr);  			if (nsPtr->parentPtr) {  			    Tcl_AppendStringsToObj(newCmd, "::", NULL); @@ -322,29 +346,29 @@ TclNamespaceEnsembleCmd(      }      case ENS_EXISTS: -	if (objc != 4) { -	    Tcl_WrongNumArgs(interp, 3, objv, "cmdname"); +	if (objc != 3) { +	    Tcl_WrongNumArgs(interp, 2, objv, "cmdname");  	    return TCL_ERROR;  	}  	Tcl_SetObjResult(interp, Tcl_NewBooleanObj( -		Tcl_FindEnsemble(interp, objv[3], 0) != NULL)); +		Tcl_FindEnsemble(interp, objv[2], 0) != NULL));  	return TCL_OK;      case ENS_CONFIG: -	if (objc < 4 || (objc != 5 && objc & 1)) { -	    Tcl_WrongNumArgs(interp, 3, objv, +	if (objc < 3 || (objc != 4 && !(objc & 1))) { +	    Tcl_WrongNumArgs(interp, 2, objv,  		    "cmdname ?-option value ...? ?arg ...?");  	    return TCL_ERROR;  	} -	token = Tcl_FindEnsemble(interp, objv[3], TCL_LEAVE_ERR_MSG); +	token = Tcl_FindEnsemble(interp, objv[2], TCL_LEAVE_ERR_MSG);  	if (token == NULL) {  	    return TCL_ERROR;  	} -	if (objc == 5) { +	if (objc == 4) {  	    Tcl_Obj *resultObj = NULL;		/* silence gcc 4 warning */ -	    if (Tcl_GetIndexFromObj(interp, objv[4], ensembleConfigOptions, +	    if (Tcl_GetIndexFromObj(interp, objv[3], ensembleConfigOptions,  		    "option", 0, &index) != TCL_OK) {  		return TCL_ERROR;  	    } @@ -370,8 +394,7 @@ TclNamespaceEnsembleCmd(  	    case CONF_NAMESPACE:  		namespacePtr = NULL;		/* silence gcc 4 warning */  		Tcl_GetEnsembleNamespace(NULL, token, &namespacePtr); -		Tcl_SetResult(interp, ((Namespace *) namespacePtr)->fullName, -			TCL_VOLATILE); +		Tcl_SetObjResult(interp, NewNsObj(namespacePtr));  		break;  	    case CONF_PREFIX: {  		int flags = 0;			/* silence gcc 4 warning */ @@ -388,7 +411,7 @@ TclNamespaceEnsembleCmd(  		}  		break;  	    } -	} else if (objc == 4) { +	} else if (objc == 3) {  	    /*  	     * Produce list of all information.  	     */ @@ -411,9 +434,7 @@ TclNamespaceEnsembleCmd(  			    -1));  	    namespacePtr = NULL;		/* silence gcc 4 warning */  	    Tcl_GetEnsembleNamespace(NULL, token, &namespacePtr); -	    Tcl_ListObjAppendElement(NULL, resultObj, -		    Tcl_NewStringObj(((Namespace *) namespacePtr)->fullName, -			    -1)); +	    Tcl_ListObjAppendElement(NULL, resultObj, NewNsObj(namespacePtr));  	    /* -parameters option */  	    Tcl_ListObjAppendElement(NULL, resultObj, @@ -457,8 +478,8 @@ TclNamespaceEnsembleCmd(  	    Tcl_GetEnsembleFlags(NULL, token, &flags);  	    permitPrefix = (flags & TCL_ENSEMBLE_PREFIX) != 0; -	    objv += 4; -	    objc -= 4; +	    objv += 3; +	    objc -= 3;  	    /*  	     * Parse the option list, applying type checks as we go. Note that @@ -515,9 +536,11 @@ TclNamespaceEnsembleCmd(  			    goto freeMapAndError;  			}  			if (len < 1) { -			    Tcl_SetResult(interp, +			    Tcl_SetObjResult(interp, Tcl_NewStringObj(  				    "ensemble subcommand implementations " -				    "must be non-empty lists", TCL_STATIC); +				    "must be non-empty lists", -1)); +			    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", +				    "EMPTY_TARGET", NULL);  			    Tcl_DictObjDone(&search);  			    if (patchedDict) {  				Tcl_DecrRefCount(patchedDict); @@ -527,8 +550,7 @@ TclNamespaceEnsembleCmd(  			cmd = TclGetString(listv[0]);  			if (!(cmd[0] == ':' && cmd[1] == ':')) {  			    Tcl_Obj *newList = Tcl_DuplicateObj(listObj); -			    Tcl_Obj *newCmd = -				    Tcl_NewStringObj(nsPtr->fullName, -1); +			    Tcl_Obj *newCmd = NewNsObj((Tcl_Namespace*)nsPtr);  			    if (nsPtr->parentPtr) {  				Tcl_AppendStringsToObj(newCmd, "::", NULL); @@ -554,7 +576,9 @@ TclNamespaceEnsembleCmd(  		    continue;  		}  		case CONF_NAMESPACE: -		    Tcl_AppendResult(interp, "option -namespace is read-only", +		    Tcl_SetObjResult(interp, Tcl_NewStringObj( +			    "option -namespace is read-only", -1)); +		    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "READ_ONLY",  			    NULL);  		    goto freeMapAndError;  		case CONF_PREFIX: @@ -616,8 +640,7 @@ Tcl_CreateEnsemble(      int flags)  {      Namespace *nsPtr = (Namespace *) namespacePtr; -    EnsembleConfig *ensemblePtr = (EnsembleConfig *) -	    ckalloc(sizeof(EnsembleConfig)); +    EnsembleConfig *ensemblePtr = ckalloc(sizeof(EnsembleConfig));      Tcl_Obj *nameObj = NULL;      if (nsPtr == NULL) { @@ -630,7 +653,7 @@ Tcl_CreateEnsemble(       */      if (!(name[0] == ':' && name[1] == ':')) { -	nameObj = Tcl_NewStringObj(nsPtr->fullName, -1); +	nameObj = NewNsObj((Tcl_Namespace *) nsPtr);  	if (nsPtr->parentPtr == NULL) {  	    Tcl_AppendStringsToObj(nameObj, name, NULL);  	} else { @@ -703,7 +726,9 @@ Tcl_SetEnsembleSubcommandList(      Tcl_Obj *oldList;      if (cmdPtr->objProc != NsEnsembleImplementationCmd) { -	Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	Tcl_SetObjResult(interp, Tcl_NewStringObj( +		"command is not an ensemble", -1)); +	Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	return TCL_ERROR;      }      if (subcmdList != NULL) { @@ -777,7 +802,9 @@ Tcl_SetEnsembleParameterList(      int length;      if (cmdPtr->objProc != NsEnsembleImplementationCmd) { -	Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	Tcl_SetObjResult(interp, Tcl_NewStringObj( +		"command is not an ensemble", -1)); +	Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	return TCL_ERROR;      }      if (paramList == NULL) { @@ -851,7 +878,9 @@ Tcl_SetEnsembleMappingDict(      Tcl_Obj *oldDict;      if (cmdPtr->objProc != NsEnsembleImplementationCmd) { -	Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	Tcl_SetObjResult(interp, Tcl_NewStringObj( +		"command is not an ensemble", -1)); +	Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	return TCL_ERROR;      }      if (mapDict != NULL) { @@ -874,9 +903,11 @@ Tcl_SetEnsembleMappingDict(  	    }  	    bytes = TclGetString(cmdObjPtr);  	    if (bytes[0] != ':' || bytes[1] != ':') { -		Tcl_AppendResult(interp, +		Tcl_SetObjResult(interp, Tcl_NewStringObj(  			"ensemble target is not a fully-qualified command", -			NULL); +			-1)); +		Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", +			"UNQUALIFIED_TARGET", NULL);  		Tcl_DictObjDone(&search);  		return TCL_ERROR;  	    } @@ -946,7 +977,9 @@ Tcl_SetEnsembleUnknownHandler(      Tcl_Obj *oldList;      if (cmdPtr->objProc != NsEnsembleImplementationCmd) { -	Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	Tcl_SetObjResult(interp, Tcl_NewStringObj( +		"command is not an ensemble", -1)); +	Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	return TCL_ERROR;      }      if (unknownList != NULL) { @@ -1010,7 +1043,9 @@ Tcl_SetEnsembleFlags(      int wasCompiled;      if (cmdPtr->objProc != NsEnsembleImplementationCmd) { -	Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	Tcl_SetObjResult(interp, Tcl_NewStringObj( +		"command is not an ensemble", -1)); +	Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	return TCL_ERROR;      } @@ -1085,7 +1120,9 @@ Tcl_GetEnsembleSubcommandList(      if (cmdPtr->objProc != NsEnsembleImplementationCmd) {  	if (interp != NULL) { -	    Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "command is not an ensemble", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	}  	return TCL_ERROR;      } @@ -1125,7 +1162,9 @@ Tcl_GetEnsembleParameterList(      if (cmdPtr->objProc != NsEnsembleImplementationCmd) {  	if (interp != NULL) { -	    Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "command is not an ensemble", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	}  	return TCL_ERROR;      } @@ -1165,7 +1204,9 @@ Tcl_GetEnsembleMappingDict(      if (cmdPtr->objProc != NsEnsembleImplementationCmd) {  	if (interp != NULL) { -	    Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "command is not an ensemble", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	}  	return TCL_ERROR;      } @@ -1204,7 +1245,9 @@ Tcl_GetEnsembleUnknownHandler(      if (cmdPtr->objProc != NsEnsembleImplementationCmd) {  	if (interp != NULL) { -	    Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "command is not an ensemble", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	}  	return TCL_ERROR;      } @@ -1243,7 +1286,9 @@ Tcl_GetEnsembleFlags(      if (cmdPtr->objProc != NsEnsembleImplementationCmd) {  	if (interp != NULL) { -	    Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "command is not an ensemble", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	}  	return TCL_ERROR;      } @@ -1282,7 +1327,9 @@ Tcl_GetEnsembleNamespace(      if (cmdPtr->objProc != NsEnsembleImplementationCmd) {  	if (interp != NULL) { -	    Tcl_AppendResult(interp, "command is not an ensemble", NULL); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "command is not an ensemble", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "NOT_ENSEMBLE", NULL);  	}  	return TCL_ERROR;      } @@ -1338,8 +1385,9 @@ Tcl_FindEnsemble(  	if (cmdPtr == NULL || cmdPtr->objProc != NsEnsembleImplementationCmd){  	    if (flags & TCL_LEAVE_ERR_MSG) { -		Tcl_AppendResult(interp, "\"", TclGetString(cmdNameObj), -			"\" is not an ensemble command", NULL); +		Tcl_SetObjResult(interp, Tcl_ObjPrintf( +			"\"%s\" is not an ensemble command", +			TclGetString(cmdNameObj)));  		Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ENSEMBLE",  			TclGetString(cmdNameObj), NULL);  	    } @@ -1426,9 +1474,9 @@ TclMakeEnsemble(      Tcl_DStringInit(&buf);      Tcl_DStringInit(&hiddenBuf); -    Tcl_DStringAppend(&hiddenBuf, "tcl:", -1); +    TclDStringAppendLiteral(&hiddenBuf, "tcl:");      Tcl_DStringAppend(&hiddenBuf, name, -1); -    Tcl_DStringAppend(&hiddenBuf, ":", -1); +    TclDStringAppendLiteral(&hiddenBuf, ":");      hiddenLen = Tcl_DStringLength(&hiddenBuf);      if (name[0] == ':' && name[1] == ':') {  	/* @@ -1444,14 +1492,14 @@ TclMakeEnsemble(  	 * multi-word list differently to a single word.  	 */ -	Tcl_DStringAppend(&buf, "::tcl", -1); +	TclDStringAppendLiteral(&buf, "::tcl");  	if (Tcl_SplitList(NULL, name, &nameCount, &nameParts) != TCL_OK) {  	    Tcl_Panic("invalid ensemble name '%s'", name);  	}  	for (i = 0; i < nameCount; ++i) { -	    Tcl_DStringAppend(&buf, "::", 2); +	    TclDStringAppendLiteral(&buf, "::");  	    Tcl_DStringAppend(&buf, nameParts[i], -1);  	}      } @@ -1476,6 +1524,14 @@ TclMakeEnsemble(  	    cmdName = nameParts[nameCount - 1];  	}      } + +    /* +     * Switch on compilation always for core ensembles now that we can do +     * nice bytecode things with them.  Do it now.  Waiting until later will +     * just cause pointless epoch bumps. +     */ + +    ensembleFlags |= ENSEMBLE_COMPILE;      ensemble = Tcl_CreateEnsemble(interp, cmdName, ns, ensembleFlags);      /* @@ -1486,7 +1542,7 @@ TclMakeEnsemble(  	Tcl_Obj *mapDict, *fromObj, *toObj;  	Command *cmdPtr; -	Tcl_DStringAppend(&buf, "::", 2); +	TclDStringAppendLiteral(&buf, "::");  	TclNewObj(mapDict);  	for (i=0 ; map[i].name != NULL ; i++) {  	    fromObj = Tcl_NewStringObj(map[i].name, -1); @@ -1524,21 +1580,15 @@ TclMakeEnsemble(  			    NULL);  		}  		cmdPtr->compileProc = map[i].compileProc; -		if (map[i].compileProc != NULL) { -		    ensembleFlags |= ENSEMBLE_COMPILE; -		}  	    }  	}  	Tcl_SetEnsembleMappingDict(interp, ensemble, mapDict); -	if (ensembleFlags & ENSEMBLE_COMPILE) { -	    Tcl_SetEnsembleFlags(interp, ensemble, ensembleFlags); -	}      }      Tcl_DStringFree(&buf);      Tcl_DStringFree(&hiddenBuf);      if (nameParts != NULL) { -	Tcl_Free((char *) nameParts); +	ckfree((char *) nameParts);      }      return ensemble;  } @@ -1592,6 +1642,7 @@ NsEnsembleImplementationCmdNR(  				 * specified but not yet cached command  				 * names. */      int reparseCount = 0;	/* Number of reparses. */ +    Tcl_Obj *errorObj;		/* Used for building error messages. */      /*       * Must recheck objc, since numParameters might have changed. Cf. test @@ -1616,10 +1667,10 @@ NsEnsembleImplementationCmdNR(  	    Tcl_Panic("List of ensemble parameters is not a list");  	}  	for (; len>0; len--,elemPtrs++) { -	    Tcl_DStringAppend(&buf, Tcl_GetString(*elemPtrs), -1); -	    Tcl_DStringAppend(&buf, " ", -1); +	    TclDStringAppendObj(&buf, *elemPtrs); +	    TclDStringAppendLiteral(&buf, " ");  	} -	Tcl_DStringAppend(&buf, "subcommand ?arg ...?", -1); +	TclDStringAppendLiteral(&buf, "subcommand ?arg ...?");  	Tcl_WrongNumArgs(interp, 1, objv, Tcl_DStringValue(&buf));  	Tcl_DStringFree(&buf); @@ -1632,8 +1683,9 @@ NsEnsembleImplementationCmdNR(  	 */  	if (!Tcl_InterpDeleted(interp)) { -	    Tcl_AppendResult(interp, -		    "ensemble activated for deleted namespace", NULL); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "ensemble activated for deleted namespace", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "DEAD", NULL);  	}  	return TCL_ERROR;      } @@ -1653,7 +1705,7 @@ NsEnsembleImplementationCmdNR(  	if (objv[1+ensemblePtr->numParameters]->typePtr==&tclEnsembleCmdType){  	    EnsembleCmdRep *ensembleCmd = objv[1+ensemblePtr->numParameters] -		    ->internalRep.otherValuePtr; +		    ->internalRep.twoPtrValue.ptr1;  	    if (ensembleCmd->nsPtr == ensemblePtr->nsPtr &&  		    ensembleCmd->epoch == ensemblePtr->epoch && @@ -1824,11 +1876,6 @@ NsEnsembleImplementationCmdNR(  	 * count both as inserted and removed arguments.  	 */ -#if 0 -	if (TclInitRewriteEnsemble(interp, 2 + ensemblePtr->numParameters, prefixObjc + ensemblePtr->numParameters, objv)) { -	    TclNRAddCallback(interp, TclClearRootEnsemble, NULL, NULL, NULL, NULL); -	} -#else  	if (iPtr->ensembleRewrite.sourceObjs == NULL) {  	    iPtr->ensembleRewrite.sourceObjs = objv;  	    iPtr->ensembleRewrite.numRemovedObjs = @@ -1849,14 +1896,13 @@ NsEnsembleImplementationCmdNR(  		iPtr->ensembleRewrite.numInsertedObjs += prefixObjc-2;  	    }  	} -#endif  	/*  	 * Hand off to the target command.  	 */ -	iPtr->evalFlags |= TCL_EVAL_REDIRECT; -	return Tcl_NREvalObj(interp, copyPtr, TCL_EVAL_INVOKE); +	TclSkipTailcall(interp); +	return TclNREvalObjEx(interp, copyPtr, TCL_EVAL_INVOKE, NULL,INT_MIN);      }    unknownOrAmbiguousSubcommand: @@ -1887,35 +1933,34 @@ NsEnsembleImplementationCmdNR(       */      Tcl_ResetResult(interp); -    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "ENSEMBLE", +    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "SUBCOMMAND",  	    TclGetString(objv[1+ensemblePtr->numParameters]), NULL);      if (ensemblePtr->subcommandTable.numEntries == 0) { -	Tcl_AppendResult(interp, "unknown subcommand \"", +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +		"unknown subcommand \"%s\": namespace %s does not" +		" export any commands",  		TclGetString(objv[1+ensemblePtr->numParameters]), -		"\": namespace ", ensemblePtr->nsPtr->fullName, -		" does not export any commands", NULL); +		ensemblePtr->nsPtr->fullName));  	Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "SUBCOMMAND",  		TclGetString(objv[1+ensemblePtr->numParameters]), NULL);  	return TCL_ERROR;      } -    Tcl_AppendResult(interp, "unknown ", -	    (ensemblePtr->flags & TCL_ENSEMBLE_PREFIX ? "or ambiguous " : ""), -	    "subcommand \"", TclGetString(objv[1+ensemblePtr->numParameters]), -	    "\": must be ", NULL); +    errorObj = Tcl_ObjPrintf("unknown%s subcommand \"%s\": must be ", +	    (ensemblePtr->flags & TCL_ENSEMBLE_PREFIX ? " or ambiguous" : ""), +	    TclGetString(objv[1+ensemblePtr->numParameters]));      if (ensemblePtr->subcommandTable.numEntries == 1) { -	Tcl_AppendResult(interp, ensemblePtr->subcommandArrayPtr[0], NULL); +	Tcl_AppendToObj(errorObj, ensemblePtr->subcommandArrayPtr[0], -1);      } else {  	int i;  	for (i=0 ; i<ensemblePtr->subcommandTable.numEntries-1 ; i++) { -	    Tcl_AppendResult(interp, -		    ensemblePtr->subcommandArrayPtr[i], ", ", NULL); +	    Tcl_AppendToObj(errorObj, ensemblePtr->subcommandArrayPtr[i], -1); +	    Tcl_AppendToObj(errorObj, ", ", 2);  	} -	Tcl_AppendResult(interp, "or ", -		ensemblePtr->subcommandArrayPtr[i], NULL); +	Tcl_AppendPrintfToObj(errorObj, "or %s", +		ensemblePtr->subcommandArrayPtr[i]);      } -    Tcl_SetErrorCode(interp, "TCL", "LOOKUP", "SUBCOMMAND", -	    TclGetString(objv[1+ensemblePtr->numParameters]), NULL); +    Tcl_SetObjResult(interp, errorObj);      return TCL_ERROR;  } @@ -2041,7 +2086,6 @@ EnsembleUnknownCallback(  {      int paramc, i, result, prefixObjc;      Tcl_Obj **paramv, *unknownCmd, *ensObj; -    char buf[TCL_INTEGER_SPACE];      /*       * Create the unknown command callback to determine what to do. @@ -2065,12 +2109,15 @@ EnsembleUnknownCallback(       */      Tcl_Preserve(ensemblePtr); -    ((Interp *) interp)->evalFlags |= TCL_EVAL_REDIRECT; +    TclSkipTailcall(interp);      result = Tcl_EvalObjv(interp, paramc, paramv, 0);      if ((result == TCL_OK) && (ensemblePtr->flags & ENSEMBLE_DEAD)) { -	Tcl_SetResult(interp, -		"unknown subcommand handler deleted its ensemble", -		TCL_STATIC); +	if (!Tcl_InterpDeleted(interp)) { +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "unknown subcommand handler deleted its ensemble", -1)); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "UNKNOWN_DELETED", +		    NULL); +	}  	result = TCL_ERROR;      }      Tcl_Release(ensemblePtr); @@ -2119,26 +2166,26 @@ EnsembleUnknownCallback(      if (!Tcl_InterpDeleted(interp)) {  	if (result != TCL_ERROR) {  	    Tcl_ResetResult(interp); -	    Tcl_SetResult(interp, -		    "unknown subcommand handler returned bad code: ", -		    TCL_STATIC); +	    Tcl_SetObjResult(interp, Tcl_NewStringObj( +		    "unknown subcommand handler returned bad code: ", -1));  	    switch (result) {  	    case TCL_RETURN: -		Tcl_AppendResult(interp, "return", NULL); +		Tcl_AppendToObj(Tcl_GetObjResult(interp), "return", -1);  		break;  	    case TCL_BREAK: -		Tcl_AppendResult(interp, "break", NULL); +		Tcl_AppendToObj(Tcl_GetObjResult(interp), "break", -1);  		break;  	    case TCL_CONTINUE: -		Tcl_AppendResult(interp, "continue", NULL); +		Tcl_AppendToObj(Tcl_GetObjResult(interp), "continue", -1);  		break;  	    default: -		sprintf(buf, "%d", result); -		Tcl_AppendResult(interp, buf, NULL); +		Tcl_AppendPrintfToObj(Tcl_GetObjResult(interp), "%d", result);  	    }  	    Tcl_AddErrorInfo(interp, "\n    result of "  		    "ensemble unknown subcommand handler: "); -	    Tcl_AddErrorInfo(interp, TclGetString(unknownCmd)); +	    Tcl_AppendObjToErrorInfo(interp, unknownCmd); +	    Tcl_SetErrorCode(interp, "TCL", "ENSEMBLE", "UNKNOWN_RESULT", +		    NULL);  	} else {  	    Tcl_AddErrorInfo(interp,  		    "\n    (ensemble unknown subcommand handler)"); @@ -2178,7 +2225,7 @@ MakeCachedEnsembleCommand(      int length;      if (objPtr->typePtr == &tclEnsembleCmdType) { -	ensembleCmd = objPtr->internalRep.otherValuePtr; +	ensembleCmd = objPtr->internalRep.twoPtrValue.ptr1;  	Tcl_DecrRefCount(ensembleCmd->realPrefixObj);  	TclNsDecrRefCount(ensembleCmd->nsPtr);  	ckfree(ensembleCmd->fullSubcmdName); @@ -2189,8 +2236,8 @@ MakeCachedEnsembleCommand(  	 */  	TclFreeIntRep(objPtr); -	ensembleCmd = (EnsembleCmdRep *) ckalloc(sizeof(EnsembleCmdRep)); -	objPtr->internalRep.otherValuePtr = ensembleCmd; +	ensembleCmd = ckalloc(sizeof(EnsembleCmdRep)); +	objPtr->internalRep.twoPtrValue.ptr1 = ensembleCmd;  	objPtr->typePtr = &tclEnsembleCmdType;      } @@ -2204,7 +2251,7 @@ MakeCachedEnsembleCommand(      ensemblePtr->nsPtr->refCount++;      ensembleCmd->realPrefixObj = prefixObjPtr;      length = strlen(subcommandName)+1; -    ensembleCmd->fullSubcmdName = ckalloc((unsigned) length); +    ensembleCmd->fullSubcmdName = ckalloc(length);      memcpy(ensembleCmd->fullSubcmdName, subcommandName, (unsigned) length);      Tcl_IncrRefCount(ensembleCmd->realPrefixObj);  } @@ -2271,7 +2318,7 @@ DeleteEnsembleConfig(       */      if (ensemblePtr->subcommandTable.numEntries != 0) { -	ckfree((char *) ensemblePtr->subcommandArrayPtr); +	ckfree(ensemblePtr->subcommandArrayPtr);      }      hEnt = Tcl_FirstHashEntry(&ensemblePtr->subcommandTable, &search);      while (hEnt != NULL) { @@ -2342,7 +2389,7 @@ BuildEnsembleConfig(  	 * Remove pre-existing table.  	 */ -	ckfree((char *) ensemblePtr->subcommandArrayPtr); +	ckfree(ensemblePtr->subcommandArrayPtr);  	hPtr = Tcl_FirstHashEntry(hash, &search);  	while (hPtr != NULL) {  	    Tcl_Obj *prefixObj = Tcl_GetHashValue(hPtr); @@ -2399,7 +2446,7 @@ BuildEnsembleConfig(  	     * the programmer's responsibility (or [::unknown] of course).  	     */ -	    cmdObj = Tcl_NewStringObj(ensemblePtr->nsPtr->fullName, -1); +	    cmdObj = NewNsObj((Tcl_Namespace *) ensemblePtr->nsPtr);  	    if (ensemblePtr->nsPtr->parentPtr != NULL) {  		Tcl_AppendStringsToObj(cmdObj, "::", name, NULL);  	    } else { @@ -2497,7 +2544,7 @@ BuildEnsembleConfig(       * the hash too, and vice versa) and running quicksort over the array.       */ -    ensemblePtr->subcommandArrayPtr = (char **) +    ensemblePtr->subcommandArrayPtr =  	    ckalloc(sizeof(char *) * hash->numEntries);      /* @@ -2585,12 +2632,12 @@ static void  FreeEnsembleCmdRep(      Tcl_Obj *objPtr)  { -    EnsembleCmdRep *ensembleCmd = objPtr->internalRep.otherValuePtr; +    EnsembleCmdRep *ensembleCmd = objPtr->internalRep.twoPtrValue.ptr1;      Tcl_DecrRefCount(ensembleCmd->realPrefixObj);      ckfree(ensembleCmd->fullSubcmdName);      TclNsDecrRefCount(ensembleCmd->nsPtr); -    ckfree((char *) ensembleCmd); +    ckfree(ensembleCmd);      objPtr->typePtr = NULL;  } @@ -2617,20 +2664,19 @@ DupEnsembleCmdRep(      Tcl_Obj *objPtr,      Tcl_Obj *copyPtr)  { -    EnsembleCmdRep *ensembleCmd = objPtr->internalRep.otherValuePtr; -    EnsembleCmdRep *ensembleCopy = (EnsembleCmdRep *) -	    ckalloc(sizeof(EnsembleCmdRep)); +    EnsembleCmdRep *ensembleCmd = objPtr->internalRep.twoPtrValue.ptr1; +    EnsembleCmdRep *ensembleCopy = ckalloc(sizeof(EnsembleCmdRep));      int length = strlen(ensembleCmd->fullSubcmdName);      copyPtr->typePtr = &tclEnsembleCmdType; -    copyPtr->internalRep.otherValuePtr = ensembleCopy; +    copyPtr->internalRep.twoPtrValue.ptr1 = ensembleCopy;      ensembleCopy->nsPtr = ensembleCmd->nsPtr;      ensembleCopy->epoch = ensembleCmd->epoch;      ensembleCopy->token = ensembleCmd->token;      ensembleCopy->nsPtr->refCount++;      ensembleCopy->realPrefixObj = ensembleCmd->realPrefixObj;      Tcl_IncrRefCount(ensembleCopy->realPrefixObj); -    ensembleCopy->fullSubcmdName = ckalloc((unsigned) length+1); +    ensembleCopy->fullSubcmdName = ckalloc(length + 1);      memcpy(ensembleCopy->fullSubcmdName, ensembleCmd->fullSubcmdName,  	    (unsigned) length+1);  } @@ -2656,11 +2702,11 @@ static void  StringOfEnsembleCmdRep(      Tcl_Obj *objPtr)  { -    EnsembleCmdRep *ensembleCmd = objPtr->internalRep.otherValuePtr; +    EnsembleCmdRep *ensembleCmd = objPtr->internalRep.twoPtrValue.ptr1;      int length = strlen(ensembleCmd->fullSubcmdName);      objPtr->length = length; -    objPtr->bytes = ckalloc((unsigned) length+1); +    objPtr->bytes = ckalloc(length + 1);      memcpy(objPtr->bytes, ensembleCmd->fullSubcmdName, (unsigned) length+1);  } @@ -2694,25 +2740,33 @@ TclCompileEnsemble(  				 * compiled. */      CompileEnv *envPtr)		/* Holds resulting instructions. */  { -    Tcl_Token *tokenPtr; +    Tcl_Token *tokenPtr = TokenAfter(parsePtr->tokenPtr);      Tcl_Obj *mapObj, *subcmdObj, *targetCmdObj, *listObj, **elems; +    Tcl_Obj *replaced = Tcl_NewObj(), *replacement;      Tcl_Command ensemble = (Tcl_Command) cmdPtr; -    Tcl_Parse synthetic; -    int len, result, flags = 0, i; +    Command *oldCmdPtr = cmdPtr, *newCmdPtr; +    int len, result, flags = 0, i, depth = 1, invokeAnyway = 0; +    int ourResult = TCL_ERROR;      unsigned numBytes;      const char *word; -    if (parsePtr->numWords < 2) { -	return TCL_ERROR; -    } +    Tcl_IncrRefCount(replaced); + +    /* +     * This is where we return to if we are parsing multiple nested compiled +     * ensembles. [info object] is such a beast. +     */ -    tokenPtr = TokenAfter(parsePtr->tokenPtr); +  checkNextWord: +    if (parsePtr->numWords < depth + 1) { +	goto failed; +    }      if (tokenPtr->type != TCL_TOKEN_SIMPLE_WORD) {  	/*  	 * Too hard.  	 */ -	return TCL_ERROR; +	goto failed;      }      word = tokenPtr[1].start; @@ -2731,7 +2785,7 @@ TclCompileEnsemble(  	 * to proceed.  	 */ -	return TCL_ERROR; +	goto failed;      }      /* @@ -2745,7 +2799,7 @@ TclCompileEnsemble(  	 * Figuring out how to compile this has become too much. Bail out.  	 */ -	return TCL_ERROR; +	goto failed;      }      /* @@ -2768,7 +2822,7 @@ TclCompileEnsemble(  	Tcl_Obj *matchObj = NULL;  	if (Tcl_ListObjGetElements(NULL, listObj, &len, &elems) != TCL_OK) { -	    return TCL_ERROR; +	    goto failed;  	}  	for (i=0 ; i<len ; i++) {  	    str = Tcl_GetStringFromObj(elems[i], &sclen); @@ -2779,8 +2833,9 @@ TclCompileEnsemble(  		result = Tcl_DictObjGet(NULL, mapObj,elems[i], &targetCmdObj);  		if (result != TCL_OK || targetCmdObj == NULL) { -		    return TCL_ERROR; +		    goto failed;  		} +		replacement = elems[i];  		goto doneMapLookup;  	    } @@ -2796,18 +2851,19 @@ TclCompileEnsemble(  	    if ((flags & TCL_ENSEMBLE_PREFIX)  		    && strncmp(word, str, numBytes) == 0) {  		if (matchObj != NULL) { -		    return TCL_ERROR; +		    goto failed;  		}  		matchObj = elems[i];  	    }  	}  	if (matchObj == NULL) { -	    return TCL_ERROR; +	    goto failed;  	}  	result = Tcl_DictObjGet(NULL, mapObj, matchObj, &targetCmdObj);  	if (result != TCL_OK || targetCmdObj == NULL) { -	    return TCL_ERROR; +	    goto failed;  	} +	replacement = matchObj;      } else {  	Tcl_DictSearch s;  	int done, matched; @@ -2819,14 +2875,15 @@ TclCompileEnsemble(  	TclNewStringObj(subcmdObj, word, (int) numBytes);  	result = Tcl_DictObjGet(NULL, mapObj, subcmdObj, &targetCmdObj); -	TclDecrRefCount(subcmdObj);  	if (result == TCL_OK && targetCmdObj != NULL) {  	    /*  	     * Got it. Skip the fiddling around with prefixes.  	     */ +	    replacement = subcmdObj;  	    goto doneMapLookup;  	} +	TclDecrRefCount(subcmdObj);  	/*  	 * We've not literally got a valid subcommand. But maybe we have a @@ -2834,7 +2891,7 @@ TclCompileEnsemble(  	 */  	if (!(flags & TCL_ENSEMBLE_PREFIX)) { -	    return TCL_ERROR; +	    goto failed;  	}  	/* @@ -2844,6 +2901,7 @@ TclCompileEnsemble(  	Tcl_DictObjFirst(NULL, mapObj, &s, &subcmdObj, &tmpObj, &done);  	matched = 0; +	replacement = NULL;		/* Silence, fool compiler! */  	while (!done) {  	    if (strncmp(TclGetString(subcmdObj), word, numBytes) == 0) {  		if (matched++) { @@ -2854,6 +2912,7 @@ TclCompileEnsemble(  		    break;  		} +		replacement = subcmdObj;  		targetCmdObj = tmpObj;  	    }  	    Tcl_DictObjNext(&s, &subcmdObj, &tmpObj, &done); @@ -2866,7 +2925,8 @@ TclCompileEnsemble(  	 */  	if (matched != 1) { -	    return TCL_ERROR; +	    invokeAnyway = 1; +	    goto failed;  	}      } @@ -2880,87 +2940,542 @@ TclCompileEnsemble(       */    doneMapLookup: +    Tcl_ListObjAppendElement(NULL, replaced, replacement);      if (Tcl_ListObjGetElements(NULL, targetCmdObj, &len, &elems) != TCL_OK) { -	return TCL_ERROR; -    } -    if (len > 1 && Tcl_IsSafe(interp)) { -	return TCL_ERROR; +	goto failed; +    } else if (len != 1) { +	/* +	 * Note that at this point we know we can't issue any special +	 * instruction sequence as the mapping isn't one that we support at +	 * the compiled level. +	 */ + +	goto cleanup;      }      targetCmdObj = elems[0]; +    oldCmdPtr = cmdPtr;      Tcl_IncrRefCount(targetCmdObj); -    cmdPtr = (Command *) Tcl_GetCommandFromObj(interp, targetCmdObj); +    newCmdPtr = (Command *) Tcl_GetCommandFromObj(interp, targetCmdObj);      TclDecrRefCount(targetCmdObj); -    if (cmdPtr == NULL || cmdPtr->compileProc == NULL) { +    if (newCmdPtr == NULL || Tcl_IsSafe(interp) +	    || newCmdPtr->nsPtr->flags & NS_SUPPRESS_COMPILATION +	    || newCmdPtr->flags & CMD_HAS_EXEC_TRACES +	    || ((Interp *)interp)->flags & DONT_COMPILE_CMDS_INLINE) {  	/*  	 * Maps to an undefined command or a command without a compiler.  	 * Cannot compile.  	 */ -	return TCL_ERROR; +	goto cleanup;      } +    cmdPtr = newCmdPtr; +    depth++;      /* -     * Now we've done the mapping process, can now actually try to compile. -     * We do this by handing off to the subcommand's actual compiler. But to -     * do that, we have to perform some trickery to rewrite the arguments. +     * See whether we have a nested ensemble. If we do, we can go round the +     * mulberry bush again, consuming the next word.       */ -    TclParseInit(interp, NULL, 0, &synthetic); -    synthetic.numWords = parsePtr->numWords - 2 + len; -    TclGrowParseTokenArray(&synthetic, 2*len); -    synthetic.numTokens = 2*len; +    if (cmdPtr->compileProc == TclCompileEnsemble) { +	tokenPtr = TokenAfter(tokenPtr); +	ensemble = (Tcl_Command) cmdPtr; +	goto checkNextWord; +    }      /* -     * Now we have the space to work in, install something rewritten. Note -     * that we are here praying for all our might that none of these words are -     * a script; the error detection code will crash if that happens and there -     * is nothing we can do to avoid it! +     * Now we've done the mapping process, can now actually try to compile. +     * If there is a subcommand compiler and that successfully produces code, +     * we'll use that. Otherwise, we fall back to generating opcodes to do the +     * invoke at runtime.       */ -    for (i=0 ; i<len ; i++) { -	int sclen; -	const char *str = Tcl_GetStringFromObj(elems[i], &sclen); +    invokeAnyway = 1; +    if (TCL_OK == TclAttemptCompileProc(interp, parsePtr, depth, cmdPtr, +	    envPtr)) { +	ourResult = TCL_OK; +	goto cleanup; +    } -	synthetic.tokenPtr[2*i].type = TCL_TOKEN_SIMPLE_WORD; -	synthetic.tokenPtr[2*i].start = str; -	synthetic.tokenPtr[2*i].size = sclen; -	synthetic.tokenPtr[2*i].numComponents = 1; +    /* +     * Failed to do a full compile for some reason. Try to do a direct invoke +     * instead of going through the ensemble lookup process again. +     */ -	synthetic.tokenPtr[2*i+1].type = TCL_TOKEN_TEXT; -	synthetic.tokenPtr[2*i+1].start = str; -	synthetic.tokenPtr[2*i+1].size = sclen; -	synthetic.tokenPtr[2*i+1].numComponents = 0; +  failed: +    if (depth < 250) { +	if (depth > 1) { +	    if (!invokeAnyway) { +		cmdPtr = oldCmdPtr; +		depth--; +	    } +	    (void) Tcl_ListObjReplace(NULL, replaced, depth, 2, 0, NULL); +	} +	CompileToInvokedCommand(interp, parsePtr, replaced, cmdPtr, envPtr); +	ourResult = TCL_OK;      }      /* -     * Copy over the real argument tokens. +     * Release the memory we allocated. If we've got here, we've either done +     * something useful or we're in a case that we can't compile at all and +     * we're just giving up.       */ -    for (i=len; i<synthetic.numWords; i++) { -	int toCopy; +  cleanup: +    Tcl_DecrRefCount(replaced); +    return ourResult; +} -	tokenPtr = TokenAfter(tokenPtr); -	toCopy = tokenPtr->numComponents + 1; -	TclGrowParseTokenArray(&synthetic, toCopy); -	memcpy(synthetic.tokenPtr + synthetic.numTokens, tokenPtr, -		sizeof(Tcl_Token) * toCopy); -	synthetic.numTokens += toCopy; +int +TclAttemptCompileProc( +    Tcl_Interp *interp, +    Tcl_Parse *parsePtr, +    int depth, +    Command *cmdPtr, +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    int result, i; +    Tcl_Token *saveTokenPtr = parsePtr->tokenPtr; +    int savedStackDepth = envPtr->currStackDepth; +    unsigned savedCodeNext = envPtr->codeNext - envPtr->codeStart; +    DefineLineInformation; + +    if (cmdPtr->compileProc == NULL) { +	return TCL_ERROR; +    } + +    /* +     * Advance parsePtr->tokenPtr so that it points at the last subcommand. +     * This will be wrong, but it will not matter, and it will put the +     * tokens for the arguments in the right place without the needed to +     * allocate a synthetic Tcl_Parse struct, or copy tokens around. +     */ + +    for (i = 0; i < depth - 1; i++) { +	parsePtr->tokenPtr = TokenAfter(parsePtr->tokenPtr);      } +    parsePtr->numWords -= (depth - 1); + +    /* +     * Shift the line information arrays to account for different word +     * index values. +     */ + +    mapPtr->loc[eclIndex].line += (depth - 1); +    mapPtr->loc[eclIndex].next += (depth - 1);      /*       * Hand off compilation to the subcommand compiler. At last!       */ -    result = cmdPtr->compileProc(interp, &synthetic, cmdPtr, envPtr); +    result = cmdPtr->compileProc(interp, parsePtr, cmdPtr, envPtr); + +    /* +     * Undo the shift.  +     */ + +    mapPtr->loc[eclIndex].line -= (depth - 1); +    mapPtr->loc[eclIndex].next -= (depth - 1); + +    parsePtr->numWords += (depth - 1); +    parsePtr->tokenPtr = saveTokenPtr;      /* -     * Clean up if necessary. +     * If our target failed to compile, revert any data from failed partial +     * compiles.  Note that envPtr->numCommands need not be checked because +     * we avoid compiling subcommands that recursively call TclCompileScript().       */ -    Tcl_FreeParse(&synthetic); +    if (result != TCL_OK) { +	envPtr->currStackDepth = savedStackDepth; +	envPtr->codeNext = envPtr->codeStart + savedCodeNext; +#ifdef TCL_COMPILE_DEBUG +    } else { +	/* +	 * Confirm that the command compiler generated a single value on +	 * the stack as its result. This is only done in debugging mode, +	 * as it *should* be correct and normal users have no reasonable +	 * way to fix it anyway. +	 */ + +	int diff = envPtr->currStackDepth - savedStackDepth; + +	if (diff != 1) { +	    Tcl_Panic("bad stack adjustment when compiling" +		    " %.*s (was %d instead of 1)", parsePtr->tokenPtr->size, +		    parsePtr->tokenPtr->start, diff); +	} +#endif +    } +      return result;  } + +/* + * How to compile a subcommand to a _replacing_ invoke of its implementation + * command. + */ + +static void +CompileToInvokedCommand( +    Tcl_Interp *interp, +    Tcl_Parse *parsePtr, +    Tcl_Obj *replacements, +    Command *cmdPtr, +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    Tcl_Token *tokPtr; +    Tcl_Obj *objPtr, **words; +    char *bytes; +    int length, i, numWords, cmdLit; +    DefineLineInformation; + +    /* +     * Push the words of the command. Take care; the command words may be +     * scripts that have backslashes in them, and [info frame 0] can see the +     * difference. Hence the call to TclContinuationsEnterDerived... +     */ + +    Tcl_ListObjGetElements(NULL, replacements, &numWords, &words); +    for (i = 0, tokPtr = parsePtr->tokenPtr; i < parsePtr->numWords; +	    i++, tokPtr = TokenAfter(tokPtr)) { +	if (i > 0 && i < numWords+1) { +	    bytes = Tcl_GetStringFromObj(words[i-1], &length); +	    PushLiteral(envPtr, bytes, length); +	    continue; +	} + +	SetLineInformation(i); +	if (tokPtr->type == TCL_TOKEN_SIMPLE_WORD) { +	    int literal = TclRegisterNewLiteral(envPtr, +		    tokPtr[1].start, tokPtr[1].size); + +	    if (envPtr->clNext) { +		TclContinuationsEnterDerived( +			TclFetchLiteral(envPtr, literal), +			tokPtr[1].start - envPtr->source, +			envPtr->clNext); +	    } +	    TclEmitPush(literal, envPtr); +	} else { +	    CompileTokens(envPtr, tokPtr, interp); +	} +    } + +    /* +     * Push the name of the command we're actually dispatching to as part of +     * the implementation. +     */ + +    objPtr = Tcl_NewObj(); +    Tcl_GetCommandFullName(interp, (Tcl_Command) cmdPtr, objPtr); +    bytes = Tcl_GetStringFromObj(objPtr, &length); +    cmdLit = TclRegisterNewCmdLiteral(envPtr, bytes, length); +    TclSetCmdNameObj(interp, TclFetchLiteral(envPtr, cmdLit), cmdPtr); +    TclEmitPush(cmdLit, envPtr); +    TclDecrRefCount(objPtr); + +    /* +     * Do the replacing dispatch. +     */ + +    TclEmitInvoke(envPtr, INST_INVOKE_REPLACE, parsePtr->numWords,numWords+1); +} + +/* + * Helpers that do issuing of instructions for commands that "don't have + * compilers" (well, they do; these). They all work by just generating base + * code to invoke the command; they're intended for ensemble subcommands so + * that the costs of INST_INVOKE_REPLACE can be avoided where we can work out + * that they're not needed. + * + * Note that these are NOT suitable for commands where there's an argument + * that is a script, as an [info level] or [info frame] in the inner context + * can see the difference. + */ + +static int +CompileBasicNArgCommand( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    Tcl_Obj *objPtr = Tcl_NewObj(); + +    Tcl_IncrRefCount(objPtr); +    Tcl_GetCommandFullName(interp, (Tcl_Command) cmdPtr, objPtr); +    TclCompileInvocation(interp, parsePtr->tokenPtr, objPtr, +	    parsePtr->numWords, envPtr); +    Tcl_DecrRefCount(objPtr); +    return TCL_OK; +} + +int +TclCompileBasic0ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords != 1) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic1ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords != 2) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic2ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords != 3) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic3ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords != 4) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic0Or1ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords != 1 && parsePtr->numWords != 2) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic1Or2ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords != 2 && parsePtr->numWords != 3) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic2Or3ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords != 3 && parsePtr->numWords != 4) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic0To2ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords < 1 || parsePtr->numWords > 3) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasic1To3ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords < 2 || parsePtr->numWords > 4) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasicMin0ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords < 1) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasicMin1ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords < 2) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +} + +int +TclCompileBasicMin2ArgCmd( +    Tcl_Interp *interp,		/* Used for error reporting. */ +    Tcl_Parse *parsePtr,	/* Points to a parse structure for the command +				 * created by Tcl_ParseCommand. */ +    Command *cmdPtr,		/* Points to defintion of command being +				 * compiled. */ +    CompileEnv *envPtr)		/* Holds resulting instructions. */ +{ +    /* +     * Verify that the number of arguments is correct; that's the only case +     * that we know will avoid the call to Tcl_WrongNumArgs() at invoke time, +     * which is the only code that sees the shenanigans of ensemble dispatch. +     */ + +    if (parsePtr->numWords < 3) { +	return TCL_ERROR; +    } + +    return CompileBasicNArgCommand(interp, parsePtr, cmdPtr, envPtr); +}  /*   * Local Variables: | 
