diff options
Diffstat (limited to 'generic/tclResult.c')
| -rw-r--r-- | generic/tclResult.c | 253 | 
1 files changed, 186 insertions, 67 deletions
| diff --git a/generic/tclResult.c b/generic/tclResult.c index 273416d..2f2563a 100644 --- a/generic/tclResult.c +++ b/generic/tclResult.c @@ -7,8 +7,6 @@   *   * See the file "license.terms" for information on usage and redistribution of   * this file, and for a DISCLAIMER OF ALL WARRANTIES. - * - * RCS: @(#) $Id: tclResult.c,v 1.57 2010/02/24 10:45:04 dkf Exp $   */  #include "tclInt.h" @@ -19,7 +17,7 @@  enum returnKeys {      KEY_CODE,	KEY_ERRORCODE,	KEY_ERRORINFO,	KEY_ERRORLINE, -    KEY_LEVEL,	KEY_OPTIONS,	KEY_LAST +    KEY_LEVEL,	KEY_OPTIONS,	KEY_ERRORSTACK,	KEY_LAST  };  /* @@ -46,6 +44,8 @@ typedef struct InterpState {      Tcl_Obj *errorCode;      Tcl_Obj *returnOpts;      Tcl_Obj *objResult; +    Tcl_Obj *errorStack; +    int resetErrorStack;  } InterpState;  /* @@ -75,13 +75,15 @@ Tcl_SaveInterpState(      int status)			/* status code for current operation */  {      Interp *iPtr = (Interp *) interp; -    InterpState *statePtr = (InterpState *) ckalloc(sizeof(InterpState)); +    InterpState *statePtr = ckalloc(sizeof(InterpState));      statePtr->status = status;      statePtr->flags = iPtr->flags & ERR_ALREADY_LOGGED;      statePtr->returnLevel = iPtr->returnLevel;      statePtr->returnCode = iPtr->returnCode;      statePtr->errorInfo = iPtr->errorInfo; +    statePtr->errorStack = iPtr->errorStack; +    statePtr->resetErrorStack = iPtr->resetErrorStack;      if (statePtr->errorInfo) {  	Tcl_IncrRefCount(statePtr->errorInfo);      } @@ -93,6 +95,9 @@ Tcl_SaveInterpState(      if (statePtr->returnOpts) {  	Tcl_IncrRefCount(statePtr->returnOpts);      } +    if (statePtr->errorStack) { +	Tcl_IncrRefCount(statePtr->errorStack); +    }      statePtr->objResult = Tcl_GetObjResult(interp);      Tcl_IncrRefCount(statePtr->objResult);      return (Tcl_InterpState) statePtr; @@ -130,6 +135,7 @@ Tcl_RestoreInterpState(      iPtr->returnLevel = statePtr->returnLevel;      iPtr->returnCode = statePtr->returnCode; +    iPtr->resetErrorStack = statePtr->resetErrorStack;      if (iPtr->errorInfo) {  	Tcl_DecrRefCount(iPtr->errorInfo);      } @@ -144,6 +150,13 @@ Tcl_RestoreInterpState(      if (iPtr->errorCode) {  	Tcl_IncrRefCount(iPtr->errorCode);      } +    if (iPtr->errorStack) { +	Tcl_DecrRefCount(iPtr->errorStack); +    } +    iPtr->errorStack = statePtr->errorStack; +    if (iPtr->errorStack) { +	Tcl_IncrRefCount(iPtr->errorStack); +    }      if (iPtr->returnOpts) {  	Tcl_DecrRefCount(iPtr->returnOpts);      } @@ -188,8 +201,11 @@ Tcl_DiscardInterpState(      if (statePtr->returnOpts) {  	Tcl_DecrRefCount(statePtr->returnOpts);      } +    if (statePtr->errorStack) { +	Tcl_DecrRefCount(statePtr->errorStack); +    }      Tcl_DecrRefCount(statePtr->objResult); -    ckfree((char *) statePtr); +    ckfree(statePtr);  }  /* @@ -214,6 +230,7 @@ Tcl_DiscardInterpState(   *----------------------------------------------------------------------   */ +#undef Tcl_SaveResult  void  Tcl_SaveResult(      Tcl_Interp *interp,		/* Interpreter to save. */ @@ -288,6 +305,7 @@ Tcl_SaveResult(   *----------------------------------------------------------------------   */ +#undef Tcl_RestoreResult  void  Tcl_RestoreResult(      Tcl_Interp *interp,		/* Interpreter being restored. */ @@ -315,7 +333,7 @@ Tcl_RestoreResult(  	 */  	if (iPtr->appendResult != NULL) { -	    ckfree((char *) iPtr->appendResult); +	    ckfree(iPtr->appendResult);  	}  	iPtr->appendResult = statePtr->appendResult; @@ -356,6 +374,7 @@ Tcl_RestoreResult(   *----------------------------------------------------------------------   */ +#undef Tcl_DiscardResult  void  Tcl_DiscardResult(      Tcl_SavedResult *statePtr)	/* State returned by Tcl_SaveResult. */ @@ -364,12 +383,10 @@ Tcl_DiscardResult(      if (statePtr->result == statePtr->appendResult) {  	ckfree(statePtr->appendResult); +    } else if (statePtr->freeProc == TCL_DYNAMIC) { +        ckfree(statePtr->result);      } else if (statePtr->freeProc) { -	if (statePtr->freeProc == TCL_DYNAMIC) { -	    ckfree(statePtr->result); -	} else { -	    statePtr->freeProc(statePtr->result); -	} +        statePtr->freeProc(statePtr->result);      }  } @@ -412,13 +429,13 @@ Tcl_SetResult(  	int length = strlen(result);  	if (length > TCL_RESULT_SIZE) { -	    iPtr->result = ckalloc((unsigned) length+1); +	    iPtr->result = ckalloc(length + 1);  	    iPtr->freeProc = TCL_DYNAMIC;  	} else {  	    iPtr->result = iPtr->resultSpace;  	    iPtr->freeProc = 0;  	} -	strcpy(iPtr->result, result); +	memcpy(iPtr->result, result, (unsigned) length+1);      } else {  	iPtr->result = (char *) result;  	iPtr->freeProc = freeProc; @@ -569,7 +586,7 @@ Tcl_GetObjResult(       * result, then reset the string result.       */ -    if (*(iPtr->result) != 0) { +    if (iPtr->result[0] != 0) {  	ResetObjResult(iPtr);  	objResultPtr = iPtr->objResultPtr; @@ -585,7 +602,7 @@ Tcl_GetObjResult(  	    iPtr->freeProc = 0;  	}  	iPtr->result = iPtr->resultSpace; -	iPtr->resultSpace[0] = 0; +	iPtr->result[0] = 0;      }      return iPtr->objResultPtr;  } @@ -632,14 +649,14 @@ Tcl_AppendResultVA(       * calls to Tcl_GetStringResult() itself. [Patch 1041072 discussion]       */ -#ifdef USE_DIRECT_INTERP_RESULT_ACCESS +#ifdef USE_INTERP_RESULT      /*       * Ensure that the interp->result is legal so old Tcl 7.* code still       * works. There's still embarrasingly much of it about...       */      (void) Tcl_GetStringResult(interp); -#endif /* USE_DIRECT_INTERP_RESULT_ACCESS */ +#endif /* USE_INTERP_RESULT */  }  /* @@ -815,7 +832,7 @@ SetupAppendBuffer(  	} else {  	    totalSpace *= 2;  	} -	new = ckalloc((unsigned) totalSpace); +	new = ckalloc(totalSpace);  	strcpy(new, iPtr->result);  	if (iPtr->appendResult != NULL) {  	    ckfree(iPtr->appendResult); @@ -924,6 +941,7 @@ Tcl_ResetResult(  	Tcl_DecrRefCount(iPtr->errorInfo);  	iPtr->errorInfo = NULL;      } +    iPtr->resetErrorStack = 1;      iPtr->returnLevel = 1;      iPtr->returnCode = TCL_OK;      if (iPtr->returnOpts) { @@ -963,14 +981,15 @@ ResetObjResult(  	TclNewObj(objResultPtr);  	Tcl_IncrRefCount(objResultPtr);  	iPtr->objResultPtr = objResultPtr; -    } else if (objResultPtr->bytes != tclEmptyStringRep) { -	if (objResultPtr->bytes != NULL) { -	    ckfree((char *) objResultPtr->bytes); +    } else { +	if (objResultPtr->bytes != tclEmptyStringRep) { +	    if (objResultPtr->bytes) { +		ckfree(objResultPtr->bytes); +	    } +	    objResultPtr->bytes = tclEmptyStringRep; +	    objResultPtr->length = 0;  	} -	objResultPtr->bytes = tclEmptyStringRep; -	objResultPtr->length = 0;  	TclFreeIntRep(objResultPtr); -	objResultPtr->typePtr = NULL;      }  } @@ -1088,13 +1107,12 @@ Tcl_SetObjErrorCode(   *   * Tcl_GetErrorLine --   * - * Results: - * - * Side effects: + *      Returns the line number associated with the current error.   *   *----------------------------------------------------------------------   */ +#undef Tcl_GetErrorLine  int  Tcl_GetErrorLine(      Tcl_Interp *interp) @@ -1107,13 +1125,12 @@ Tcl_GetErrorLine(   *   * Tcl_SetErrorLine --   * - * Results: - * - * Side effects: + *      Sets the line number associated with the current error.   *   *----------------------------------------------------------------------   */ +#undef Tcl_SetErrorLine  void  Tcl_SetErrorLine(      Tcl_Interp *interp, @@ -1161,6 +1178,7 @@ GetKeys(void)  	TclNewLiteralStringObj(keys[KEY_ERRORCODE], "-errorcode");  	TclNewLiteralStringObj(keys[KEY_ERRORINFO], "-errorinfo");  	TclNewLiteralStringObj(keys[KEY_ERRORLINE], "-errorline"); +	TclNewLiteralStringObj(keys[KEY_ERRORSTACK],"-errorstack");  	TclNewLiteralStringObj(keys[KEY_LEVEL],	    "-level");  	TclNewLiteralStringObj(keys[KEY_OPTIONS],   "-options"); @@ -1255,7 +1273,8 @@ TclProcessReturn(  	    Tcl_DecrRefCount(iPtr->errorInfo);  	    iPtr->errorInfo = NULL;  	} -	Tcl_DictObjGet(NULL, iPtr->returnOpts, keys[KEY_ERRORINFO], &valuePtr); +	Tcl_DictObjGet(NULL, iPtr->returnOpts, keys[KEY_ERRORINFO], +                &valuePtr);  	if (valuePtr != NULL) {  	    int infoLen; @@ -1266,14 +1285,50 @@ TclProcessReturn(  		iPtr->flags |= ERR_ALREADY_LOGGED;  	    }  	} -	Tcl_DictObjGet(NULL, iPtr->returnOpts, keys[KEY_ERRORCODE], &valuePtr); +	Tcl_DictObjGet(NULL, iPtr->returnOpts, keys[KEY_ERRORSTACK], +                &valuePtr); +	if (valuePtr != NULL) { +            int len, valueObjc; +            Tcl_Obj **valueObjv; + +            if (Tcl_IsShared(iPtr->errorStack)) { +                Tcl_Obj *newObj; +                 +                newObj = Tcl_DuplicateObj(iPtr->errorStack); +                Tcl_DecrRefCount(iPtr->errorStack); +                Tcl_IncrRefCount(newObj); +                iPtr->errorStack = newObj; +            } + +            /* +             * List extraction done after duplication to avoid moving the rug +             * if someone does [return -errorstack [info errorstack]] +             */ + +            if (Tcl_ListObjGetElements(interp, valuePtr, &valueObjc, +                    &valueObjv) == TCL_ERROR) { +                return TCL_ERROR; +            } +            iPtr->resetErrorStack = 0; +            Tcl_ListObjLength(interp, iPtr->errorStack, &len); + +            /* +             * Reset while keeping the list intrep as much as possible. +             */ + +            Tcl_ListObjReplace(interp, iPtr->errorStack, 0, len, valueObjc, +                    valueObjv); + 	} +	Tcl_DictObjGet(NULL, iPtr->returnOpts, keys[KEY_ERRORCODE], +                &valuePtr);  	if (valuePtr != NULL) {  	    Tcl_SetObjErrorCode(interp, valuePtr);  	} else {  	    Tcl_SetErrorCode(interp, "NONE", NULL);  	} -	Tcl_DictObjGet(NULL, iPtr->returnOpts, keys[KEY_ERRORLINE], &valuePtr); +	Tcl_DictObjGet(NULL, iPtr->returnOpts, keys[KEY_ERRORLINE], +                &valuePtr);  	if (valuePtr != NULL) {  	    TclGetIntFromObj(NULL, valuePtr, &iPtr->errorLine);  	} @@ -1297,7 +1352,7 @@ TclProcessReturn(   *	Parses, checks, and stores the options to the [return] command.   *   * Results: - *	Returns TCL_ERROR is any of the option values are invalid. Otherwise, + *	Returns TCL_ERROR if any of the option values are invalid. Otherwise,   *	returns TCL_OK, and writes the returnOpts, code, and level values to   *	the pointers provided.   * @@ -1333,7 +1388,7 @@ TclMergeReturnOptions(  	const char *compare =  		TclGetStringFromObj(keys[KEY_OPTIONS], &compareLen); -	if ((optLen == compareLen) && (strcmp(opt, compare) == 0)) { +	if ((optLen == compareLen) && (memcmp(opt, compare, optLen) == 0)) {  	    Tcl_DictSearch search;  	    int done = 0;  	    Tcl_Obj *keyPtr; @@ -1346,10 +1401,9 @@ TclMergeReturnOptions(  		 * Value is not a legal dictionary.  		 */ -		Tcl_ResetResult(interp); -		Tcl_AppendResult(interp, "bad ", compare, -			" value: expected dictionary but got \"", -			TclGetString(objv[1]), "\"", NULL); +		Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                        "bad %s value: expected dictionary but got \"%s\"", +                        compare, TclGetString(objv[1])));  		Tcl_SetErrorCode(interp, "TCL", "RESULT", "ILLEGAL_OPTIONS",  			NULL);  		goto error; @@ -1377,28 +1431,11 @@ TclMergeReturnOptions(       */      Tcl_DictObjGet(NULL, returnOpts, keys[KEY_CODE], &valuePtr); -    if ((valuePtr != NULL) -	    && (TCL_ERROR == TclGetIntFromObj(NULL, valuePtr, &code))) { -	static const char *const returnCodes[] = { -	    "ok", "error", "return", "break", "continue", NULL -	}; - -	if (TCL_ERROR == Tcl_GetIndexFromObj(NULL, valuePtr, returnCodes, -		NULL, TCL_EXACT, &code)) { -	    /* -	     * Value is not a legal return code. -	     */ - -	    Tcl_ResetResult(interp); -	    Tcl_AppendResult(interp, "bad completion code \"", -		    TclGetString(valuePtr), -		    "\": must be ok, error, return, break, " -		    "continue, or an integer", NULL); -	    Tcl_SetErrorCode(interp, "TCL", "RESULT", "ILLEGAL_CODE", NULL); +    if (valuePtr != NULL) { +	if (TclGetCompletionCodeFromObj(interp, valuePtr, +                &code) == TCL_ERROR) {  	    goto error;  	} -    } -    if (valuePtr != NULL) {  	Tcl_DictObjRemove(NULL, returnOpts, keys[KEY_CODE]);      } @@ -1414,10 +1451,9 @@ TclMergeReturnOptions(  	     * Value is not a legal level.  	     */ -	    Tcl_ResetResult(interp); -	    Tcl_AppendResult(interp, "bad -level value: " -		    "expected non-negative integer but got \"", -		    TclGetString(valuePtr), "\"", NULL); +	    Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                    "bad -level value: expected non-negative integer but got" +                    " \"%s\"", TclGetString(valuePtr)));  	    Tcl_SetErrorCode(interp, "TCL", "RESULT", "ILLEGAL_LEVEL", NULL);  	    goto error;  	} @@ -1425,6 +1461,62 @@ TclMergeReturnOptions(      }      /* +     * Check for bogus -errorcode value. +     */ + +    Tcl_DictObjGet(NULL, returnOpts, keys[KEY_ERRORCODE], &valuePtr); +    if (valuePtr != NULL) { +	int length; + +	if (TCL_ERROR == Tcl_ListObjLength(NULL, valuePtr, &length )) { +	    /* +	     * Value is not a list, which is illegal for -errorcode. +	     */ + +	    Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                    "bad -errorcode value: expected a list but got \"%s\"", +                    TclGetString(valuePtr))); +	    Tcl_SetErrorCode(interp, "TCL", "RESULT", "ILLEGAL_ERRORCODE", +		    NULL); +	    goto error; +	} +    } + +    /* +     * Check for bogus -errorstack value. +     */ + +    Tcl_DictObjGet(NULL, returnOpts, keys[KEY_ERRORSTACK], &valuePtr); +    if (valuePtr != NULL) { +	int length; + +	if (TCL_ERROR == Tcl_ListObjLength(NULL, valuePtr, &length )) { +	    /* +	     * Value is not a list, which is illegal for -errorstack. +	     */ + +	    Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                    "bad -errorstack value: expected a list but got \"%s\"", +                    TclGetString(valuePtr))); +	    Tcl_SetErrorCode(interp, "TCL", "RESULT", "NONLIST_ERRORSTACK", +                    NULL); +	    goto error; +	} +        if (length % 2) { +            /* +             * Errorstack must always be an even-sized list +             */ + +	    Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                    "forbidden odd-sized list for -errorstack: \"%s\"", +		    TclGetString(valuePtr))); +	    Tcl_SetErrorCode(interp, "TCL", "RESULT", +                    "ODDSIZEDLIST_ERRORSTACK", NULL); +	    goto error; +        } +    } + +    /*       * Convert [return -code return -level X] to [return -code ok -level X+1]       */ @@ -1500,7 +1592,8 @@ Tcl_GetReturnOptions(      }      if (result == TCL_ERROR) { -	Tcl_AddObjErrorInfo(interp, "", -1); +	Tcl_AddErrorInfo(interp, ""); +        Tcl_DictObjPut(NULL, options, keys[KEY_ERRORSTACK], iPtr->errorStack);      }      if (iPtr->errorCode) {  	Tcl_DictObjPut(NULL, options, keys[KEY_ERRORCODE], iPtr->errorCode); @@ -1516,6 +1609,31 @@ Tcl_GetReturnOptions(  /*   *-------------------------------------------------------------------------   * + * TclNoErrorStack -- + * + *	Removes the -errorstack entry from an options dict to avoid reference + *	cycles. + * + * Results: + *	The (unshared) argument options dict, modified in -place. + * + *------------------------------------------------------------------------- + */ + +Tcl_Obj * +TclNoErrorStack( +    Tcl_Interp *interp, +    Tcl_Obj *options) +{ +    Tcl_Obj **keys = GetKeys(); +     +    Tcl_DictObjRemove(interp, options, keys[KEY_ERRORSTACK]); +    return options; +} + +/* + *------------------------------------------------------------------------- + *   * Tcl_SetReturnOptions --   *   *	Accepts an interp and a dictionary of return options, and sets the @@ -1544,9 +1662,8 @@ Tcl_SetReturnOptions(      Tcl_IncrRefCount(options);      if (TCL_ERROR == TclListObjGetElements(interp, options, &objc, &objv)  	    || (objc % 2)) { -	Tcl_ResetResult(interp); -	Tcl_AppendResult(interp, "expected dict but got \"", -		TclGetString(options), "\"", NULL); +	Tcl_SetObjResult(interp, Tcl_ObjPrintf( +                "expected dict but got \"%s\"", TclGetString(options)));  	Tcl_SetErrorCode(interp, "TCL", "RESULT", "ILLEGAL_OPTIONS", NULL);  	code = TCL_ERROR;      } else if (TCL_ERROR == TclMergeReturnOptions(interp, objc, objv, @@ -1632,5 +1749,7 @@ Tcl_TransferResult(   * mode: c   * c-basic-offset: 4   * fill-column: 78 + * tab-width: 8 + * indent-tabs-mode: nil   * End:   */ | 
