diff options
Diffstat (limited to 'generic/tclTest.c')
| -rw-r--r-- | generic/tclTest.c | 750 | 
1 files changed, 582 insertions, 168 deletions
| diff --git a/generic/tclTest.c b/generic/tclTest.c index 5aedb14..a27c95a 100644 --- a/generic/tclTest.c +++ b/generic/tclTest.c @@ -13,8 +13,6 @@   *   * See the file "license.terms" for information on usage and redistribution of   * this file, and for a DISCLAIMER OF ALL WARRANTIES. - * - * RCS: @(#) $Id: tclTest.c,v 1.150 2010/03/11 13:35:25 nijtmans Exp $   */  #undef STATIC_BUILD @@ -23,6 +21,8 @@  #endif  #include "tclInt.h"  #include "tclOO.h" +#include <math.h> +  /*   * Required for Testregexp*Cmd   */ @@ -75,6 +75,8 @@ typedef struct TestAsyncHandler {  				/* Next is list of handlers. */  } TestAsyncHandler; +TCL_DECLARE_MUTEX(asyncTestMutex) +  static TestAsyncHandler *firstHandler = NULL;  /* @@ -233,6 +235,9 @@ static int		TestdelCmd(ClientData dummy,  			    Tcl_Interp *interp, int argc, const char **argv);  static int		TestdelassocdataCmd(ClientData dummy,  			    Tcl_Interp *interp, int argc, const char **argv); +static int		TestdoubledigitsObjCmd(ClientData dummy, +					       Tcl_Interp* interp, +					       int objc, Tcl_Obj* const objv[]);  static int		TestdstringCmd(ClientData dummy,  			    Tcl_Interp *interp, int argc, const char **argv);  static int		TestencodingObjCmd(ClientData dummy, @@ -303,9 +308,8 @@ static int		TestexitmainloopCmd(ClientData dummy,  			    Tcl_Interp *interp, int argc, const char **argv);  static int		TestpanicCmd(ClientData dummy,  			    Tcl_Interp *interp, int argc, const char **argv); -static int		TestfinexitObjCmd(ClientData dummy, -			    Tcl_Interp *interp, int objc, -			    Tcl_Obj *const objv[]); +static int		TestparseargsCmd(ClientData dummy, Tcl_Interp *interp, +			    int objc, Tcl_Obj *const objv[]);  static int		TestparserObjCmd(ClientData dummy,  			    Tcl_Interp *interp, int objc,  			    Tcl_Obj *const objv[]); @@ -323,10 +327,12 @@ static int		TestreturnObjCmd(ClientData dummy,  			    Tcl_Obj *const objv[]);  static void		TestregexpXflags(const char *string,  			    int length, int *cflagsPtr, int *eflagsPtr); +#ifndef TCL_NO_DEPRECATED  static int		TestsaveresultCmd(ClientData dummy,  			    Tcl_Interp *interp, int objc,  			    Tcl_Obj *const objv[]);  static void		TestsaveresultFree(char *blockPtr); +#endif /* TCL_NO_DEPRECATED */  static int		TestsetassocdataCmd(ClientData dummy,  			    Tcl_Interp *interp, int argc, const char **argv);  static int		TestsetCmd(ClientData dummy, @@ -365,55 +371,37 @@ static int		TestSimpleFilesystemObjCmd(  static void		TestReport(const char *cmd, Tcl_Obj *arg1,  			    Tcl_Obj *arg2);  static Tcl_Obj *	TestReportGetNativePath(Tcl_Obj *pathPtr); -static int		TestReportStat(Tcl_Obj *path, Tcl_StatBuf *buf); -static int		TestReportAccess(Tcl_Obj *path, int mode); -static Tcl_Channel	TestReportOpenFileChannel( -			    Tcl_Interp *interp, Tcl_Obj *fileName, -			    int mode, int permissions); -static int		TestReportMatchInDirectory(Tcl_Interp *interp, -			    Tcl_Obj *resultPtr, Tcl_Obj *dirPtr, -			    const char *pattern, Tcl_GlobTypeData *types); -static int		TestReportChdir(Tcl_Obj *dirName); -static int		TestReportLstat(Tcl_Obj *path, Tcl_StatBuf *buf); -static int		TestReportCopyFile(Tcl_Obj *src, Tcl_Obj *dst); -static int		TestReportDeleteFile(Tcl_Obj *path); -static int		TestReportRenameFile(Tcl_Obj *src, Tcl_Obj *dst); -static int		TestReportCreateDirectory(Tcl_Obj *path); -static int		TestReportCopyDirectory(Tcl_Obj *src, -			    Tcl_Obj *dst, Tcl_Obj **errorPtr); -static int		TestReportRemoveDirectory(Tcl_Obj *path, -			    int recursive, Tcl_Obj **errorPtr); -static int		TestReportLoadFile(Tcl_Interp *interp, -			    Tcl_Obj *fileName, Tcl_LoadHandle *handlePtr, -			    Tcl_FSUnloadFileProc **unloadProcPtr); -static Tcl_Obj *	TestReportLink(Tcl_Obj *path, -			    Tcl_Obj *to, int linkType); -static const char *const *TestReportFileAttrStrings( -			    Tcl_Obj *fileName, Tcl_Obj **objPtrRef); -static int		TestReportFileAttrsGet(Tcl_Interp *interp, -			    int index, Tcl_Obj *fileName, Tcl_Obj **objPtrRef); -static int		TestReportFileAttrsSet(Tcl_Interp *interp, -			    int index, Tcl_Obj *fileName, Tcl_Obj *objPtr); -static int		TestReportUtime(Tcl_Obj *fileName, -			    struct utimbuf *tval); -static int		TestReportNormalizePath(Tcl_Interp *interp, -			    Tcl_Obj *pathPtr, int nextCheckpoint); -static int		TestReportInFilesystem(Tcl_Obj *pathPtr, ClientData *clientDataPtr); -static void		TestReportFreeInternalRep(ClientData clientData); -static ClientData	TestReportDupInternalRep(ClientData clientData); - -static int		SimpleStat(Tcl_Obj *path, Tcl_StatBuf *buf); -static int		SimpleAccess(Tcl_Obj *path, int mode); -static Tcl_Channel	SimpleOpenFileChannel(Tcl_Interp *interp, -			    Tcl_Obj *fileName, int mode, int permissions); -static Tcl_Obj *	SimpleListVolumes(void); -static int		SimplePathInFilesystem( -			    Tcl_Obj *pathPtr, ClientData *clientDataPtr); +static Tcl_FSStatProc TestReportStat; +static Tcl_FSAccessProc TestReportAccess; +static Tcl_FSOpenFileChannelProc TestReportOpenFileChannel; +static Tcl_FSMatchInDirectoryProc TestReportMatchInDirectory; +static Tcl_FSChdirProc TestReportChdir; +static Tcl_FSLstatProc TestReportLstat; +static Tcl_FSCopyFileProc TestReportCopyFile; +static Tcl_FSDeleteFileProc TestReportDeleteFile; +static Tcl_FSRenameFileProc TestReportRenameFile; +static Tcl_FSCreateDirectoryProc TestReportCreateDirectory; +static Tcl_FSCopyDirectoryProc TestReportCopyDirectory; +static Tcl_FSRemoveDirectoryProc TestReportRemoveDirectory; +static int TestReportLoadFile(Tcl_Interp *interp, Tcl_Obj *pathPtr, +	Tcl_LoadHandle *handlePtr, Tcl_FSUnloadFileProc **unloadProcPtr); +static Tcl_FSLinkProc TestReportLink; +static Tcl_FSFileAttrStringsProc TestReportFileAttrStrings; +static Tcl_FSFileAttrsGetProc TestReportFileAttrsGet; +static Tcl_FSFileAttrsSetProc TestReportFileAttrsSet; +static Tcl_FSUtimeProc TestReportUtime; +static Tcl_FSNormalizePathProc TestReportNormalizePath; +static Tcl_FSPathInFilesystemProc TestReportInFilesystem; +static Tcl_FSFreeInternalRepProc TestReportFreeInternalRep; +static Tcl_FSDupInternalRepProc TestReportDupInternalRep; + +static Tcl_FSStatProc SimpleStat; +static Tcl_FSAccessProc SimpleAccess; +static Tcl_FSOpenFileChannelProc SimpleOpenFileChannel; +static Tcl_FSListVolumesProc SimpleListVolumes; +static Tcl_FSPathInFilesystemProc SimplePathInFilesystem;  static Tcl_Obj *	SimpleRedirect(Tcl_Obj *pathPtr); -static int		SimpleMatchInDirectory( -			    Tcl_Interp *interp, Tcl_Obj *resultPtr, -			    Tcl_Obj *dirPtr, const char *pattern, -			    Tcl_GlobTypeData *types); +static Tcl_FSMatchInDirectoryProc SimpleMatchInDirectory;  static int		TestNumUtfCharsCmd(ClientData clientData,  			    Tcl_Interp *interp, int objc,  			    Tcl_Obj *const objv[]); @@ -423,6 +411,14 @@ static int		TestHashSystemHashCmd(ClientData clientData,  static int		TestNRELevels(ClientData clientData,  			    Tcl_Interp *interp, int objc,  			    Tcl_Obj *const objv[]); +static int		TestInterpResolverCmd(ClientData clientData, +			    Tcl_Interp *interp, int objc, +			    Tcl_Obj *const objv[]); +#if defined(HAVE_CPUID) || defined(_WIN32) +static int		TestcpuidCmd(ClientData dummy, +			    Tcl_Interp* interp, int objc, +			    Tcl_Obj *const objv[]); +#endif  static const Tcl_Filesystem testReportingFilesystem = {      "reporting", @@ -453,7 +449,7 @@ static const Tcl_Filesystem testReportingFilesystem = {      TestReportRenameFile,      TestReportCopyDirectory,      TestReportLstat, -    TestReportLoadFile, +    (Tcl_FSLoadFileProc *) TestReportLoadFile,      NULL /* cwd */,      TestReportChdir  }; @@ -529,7 +525,9 @@ int  Tcltest_Init(      Tcl_Interp *interp)		/* Interpreter for application. */  { +#ifndef TCL_NO_DEPRECATED      Tcl_ValueType t3ArgTypes[2]; +#endif /* TCL_NO_DEPRECATED */      Tcl_Obj *listPtr;      Tcl_Obj **objv; @@ -588,6 +586,8 @@ Tcltest_Init(      Tcl_CreateCommand(interp, "testdel", TestdelCmd, NULL, NULL);      Tcl_CreateCommand(interp, "testdelassocdata", TestdelassocdataCmd,  	    NULL, NULL); +    Tcl_CreateObjCommand(interp, "testdoubledigits", TestdoubledigitsObjCmd, +			 NULL, NULL);      Tcl_DStringInit(&dstring);      Tcl_CreateCommand(interp, "testdstring", TestdstringCmd, NULL,  	    NULL); @@ -635,7 +635,7 @@ Tcltest_Init(      Tcl_CreateObjCommand(interp, "testlocale", TestlocaleCmd, NULL,  	    NULL);      Tcl_CreateCommand(interp, "testpanic", TestpanicCmd, NULL, NULL); -    Tcl_CreateObjCommand(interp, "testfinexit", TestfinexitObjCmd, NULL, NULL); +    Tcl_CreateObjCommand(interp, "testparseargs", TestparseargsCmd,NULL,NULL);      Tcl_CreateObjCommand(interp, "testparser", TestparserObjCmd,  	    NULL, NULL);      Tcl_CreateObjCommand(interp, "testparsevar", TestparsevarObjCmd, @@ -646,8 +646,10 @@ Tcltest_Init(  	    NULL, NULL);      Tcl_CreateObjCommand(interp, "testreturn", TestreturnObjCmd,  	    NULL, NULL); +#ifndef TCL_NO_DEPRECATED      Tcl_CreateObjCommand(interp, "testsaveresult", TestsaveresultCmd,  	    NULL, NULL); +#endif /* TCL_NO_DEPRECATED */      Tcl_CreateCommand(interp, "testsetassocdata", TestsetassocdataCmd,  	    NULL, NULL);      Tcl_CreateCommand(interp, "testsetnoerr", TestsetCmd, @@ -669,21 +671,31 @@ Tcltest_Init(      Tcl_CreateCommand(interp, "testtranslatefilename",  	    TesttranslatefilenameCmd, NULL, NULL);      Tcl_CreateCommand(interp, "testupvar", TestupvarCmd, NULL, NULL); +#ifndef TCL_NO_DEPRECATED      Tcl_CreateMathFunc(interp, "T1", 0, NULL, TestMathFunc, (ClientData) 123);      Tcl_CreateMathFunc(interp, "T2", 0, NULL, TestMathFunc, (ClientData) 345); +#endif /* TCL_NO_DEPRECATED */      Tcl_CreateCommand(interp, "testmainthread", TestmainthreadCmd, NULL,  	    NULL);      Tcl_CreateCommand(interp, "testsetmainloop", TestsetmainloopCmd,  	    NULL, NULL);      Tcl_CreateCommand(interp, "testexitmainloop", TestexitmainloopCmd,  	    NULL, NULL); +#if defined(HAVE_CPUID) || defined(_WIN32) +    Tcl_CreateObjCommand(interp, "testcpuid", TestcpuidCmd, +	    (ClientData) 0, NULL); +#endif +#ifndef TCL_NO_DEPRECATED      t3ArgTypes[0] = TCL_EITHER;      t3ArgTypes[1] = TCL_EITHER;      Tcl_CreateMathFunc(interp, "T3", 2, t3ArgTypes, TestMathFunc2,  	    NULL); +#endif /* TCL_NO_DEPRECATED */      Tcl_CreateObjCommand(interp, "testnrelevels", TestNRELevels,  	    NULL, NULL); +    Tcl_CreateObjCommand(interp, "testinterpresolver", TestInterpResolverCmd, +	    NULL, NULL);      if (TclObjTest_Init(interp) != TCL_OK) {  	return TCL_ERROR; @@ -804,25 +816,29 @@ TestasyncCmd(  	if (argc != 3) {  	    goto wrongNumArgs;  	} -	asyncPtr = (TestAsyncHandler *) ckalloc(sizeof(TestAsyncHandler)); +	asyncPtr = ckalloc(sizeof(TestAsyncHandler)); +	asyncPtr->command = ckalloc(strlen(argv[2]) + 1); +	strcpy(asyncPtr->command, argv[2]); +        Tcl_MutexLock(&asyncTestMutex);  	asyncPtr->id = nextId;  	nextId++;  	asyncPtr->handler = Tcl_AsyncCreate(AsyncHandlerProc, -		(ClientData) asyncPtr); -	asyncPtr->command = (char *) ckalloc((unsigned) (strlen(argv[2]) + 1)); -	strcpy(asyncPtr->command, argv[2]); +                                            INT2PTR(asyncPtr->id));  	asyncPtr->nextPtr = firstHandler;  	firstHandler = asyncPtr; +        Tcl_MutexUnlock(&asyncTestMutex);  	Tcl_SetObjResult(interp, Tcl_NewIntObj(asyncPtr->id));      } else if (strcmp(argv[1], "delete") == 0) {  	if (argc == 2) { +            Tcl_MutexLock(&asyncTestMutex);  	    while (firstHandler != NULL) {  		asyncPtr = firstHandler;  		firstHandler = asyncPtr->nextPtr;  		Tcl_AsyncDelete(asyncPtr->handler);  		ckfree(asyncPtr->command); -		ckfree((char *) asyncPtr); +		ckfree(asyncPtr);  	    } +            Tcl_MutexUnlock(&asyncTestMutex);  	    return TCL_OK;  	}  	if (argc != 3) { @@ -831,6 +847,7 @@ TestasyncCmd(  	if (Tcl_GetInt(interp, argv[2], &id) != TCL_OK) {  	    return TCL_ERROR;  	} +        Tcl_MutexLock(&asyncTestMutex);  	for (prevPtr = NULL, asyncPtr = firstHandler; asyncPtr != NULL;  		prevPtr = asyncPtr, asyncPtr = asyncPtr->nextPtr) {  	    if (asyncPtr->id != id) { @@ -843,9 +860,10 @@ TestasyncCmd(  	    }  	    Tcl_AsyncDelete(asyncPtr->handler);  	    ckfree(asyncPtr->command); -	    ckfree((char *) asyncPtr); +	    ckfree(asyncPtr);  	    break;  	} +        Tcl_MutexUnlock(&asyncTestMutex);      } else if (strcmp(argv[1], "mark") == 0) {  	if (argc != 5) {  	    goto wrongNumArgs; @@ -854,6 +872,7 @@ TestasyncCmd(  		|| (Tcl_GetInt(interp, argv[4], &code) != TCL_OK)) {  	    return TCL_ERROR;  	} +	Tcl_MutexLock(&asyncTestMutex);  	for (asyncPtr = firstHandler; asyncPtr != NULL;  		asyncPtr = asyncPtr->nextPtr) {  	    if (asyncPtr->id == id) { @@ -862,6 +881,7 @@ TestasyncCmd(  	    }  	}  	Tcl_SetObjResult(interp, Tcl_NewStringObj(argv[3], -1)); +	Tcl_MutexUnlock(&asyncTestMutex);  	return code;  #ifdef TCL_THREADS      } else if (strcmp(argv[1], "marklater") == 0) { @@ -871,19 +891,22 @@ TestasyncCmd(  	if (Tcl_GetInt(interp, argv[2], &id) != TCL_OK) {  	    return TCL_ERROR;  	} +        Tcl_MutexLock(&asyncTestMutex);  	for (asyncPtr = firstHandler; asyncPtr != NULL;  		asyncPtr = asyncPtr->nextPtr) {  	    if (asyncPtr->id == id) {  		Tcl_ThreadId threadID;  		if (Tcl_CreateThread(&threadID, AsyncThreadProc, -			(ClientData) asyncPtr, TCL_THREAD_STACK_DEFAULT, +			INT2PTR(id), TCL_THREAD_STACK_DEFAULT,  			TCL_THREAD_NOFLAGS) != TCL_OK) {  		    Tcl_SetResult(interp, "can't create thread", TCL_STATIC); +		    Tcl_MutexUnlock(&asyncTestMutex);  		    return TCL_ERROR;  		}  		break;  	    }  	} +        Tcl_MutexUnlock(&asyncTestMutex);      } else {  	Tcl_AppendResult(interp, "bad option \"", argv[1],  		"\": must be create, delete, int, mark, or marklater", NULL); @@ -900,15 +923,29 @@ TestasyncCmd(  static int  AsyncHandlerProc( -    ClientData clientData,	/* Pointer to TestAsyncHandler structure. */ +    ClientData clientData,	/* If of TestAsyncHandler structure.  +                                 * in global list. */      Tcl_Interp *interp,		/* Interpreter in which command was  				 * executed, or NULL. */      int code)			/* Current return code from command. */  { -    TestAsyncHandler *asyncPtr = (TestAsyncHandler *) clientData; +    TestAsyncHandler *asyncPtr; +    int id = PTR2INT(clientData);      const char *listArgv[4], *cmd;      char string[TCL_INTEGER_SPACE]; +    Tcl_MutexLock(&asyncTestMutex); +    for (asyncPtr = firstHandler; asyncPtr != NULL; +         asyncPtr = asyncPtr->nextPtr) { +        if (asyncPtr->id == id) break; +    } +    Tcl_MutexUnlock(&asyncTestMutex); + +    if (!asyncPtr) { +        /* Woops - this one was deleted between the AsyncMark and now */ +        return TCL_OK; +    } +      TclFormatInt(string, code);      listArgv[0] = asyncPtr->command;      listArgv[1] = Tcl_GetString(Tcl_GetObjResult(interp)); @@ -923,7 +960,7 @@ AsyncHandlerProc(  	 * invoked, it's possible.  Better error checking is needed here.  	 */      } -    ckfree((char *)cmd); +    ckfree(cmd);      return code;  } @@ -946,12 +983,22 @@ AsyncHandlerProc(  #ifdef TCL_THREADS  static Tcl_ThreadCreateType  AsyncThreadProc( -    ClientData clientData)	/* Parameter is a pointer to a +    ClientData clientData)	/* Parameter is the id of a  				 * TestAsyncHandler, defined above. */  { -    TestAsyncHandler *asyncPtr = clientData; +    TestAsyncHandler *asyncPtr; +    int id = PTR2INT(clientData); +      Tcl_Sleep(1); -    Tcl_AsyncMark(asyncPtr->handler); +    Tcl_MutexLock(&asyncTestMutex); +    for (asyncPtr = firstHandler; asyncPtr != NULL; +         asyncPtr = asyncPtr->nextPtr) { +        if (asyncPtr->id == id) { +            Tcl_AsyncMark(asyncPtr->handler); +            break; +        } +    } +    Tcl_MutexUnlock(&asyncTestMutex);      Tcl_ExitThread(TCL_OK);      TCL_THREAD_CREATE_RETURN;  } @@ -1201,7 +1248,7 @@ TestcmdtraceCmd(  	 * Create a command trace then eval a script to check whether it is  	 * called. Note that this trace procedure removes itself as a further  	 * check of the robustness of the trace proc calling code in -	 * TclExecuteByteCode. +	 * TclNRExecuteByteCode.  	 */  	cmdTrace = Tcl_CreateTrace(interp, 50000, CmdTraceDeleteProc, NULL); @@ -1301,7 +1348,7 @@ CmdTraceDeleteProc(  {      /*       * Remove ourselves to test whether calling Tcl_DeleteTrace within a trace -     * callback causes the for loop in TclExecuteByteCode that calls traces to +     * callback causes the for loop in TclNRExecuteByteCode that calls traces to       * reference freed memory.       */ @@ -1508,14 +1555,14 @@ DelCallbackProc(   *   * TestdelCmd --   * - *	This procedure implements the "testdcall" command.  It is used - *	to test Tcl_CallWhenDeleted. + *	This procedure implements the "testdel" command.  It is used + *	to test calling of command deletion callbacks.   *   * Results:   *	A standard Tcl result.   *   * Side effects: - *	Creates and deletes interpreters. + *	Creates a command.   *   *----------------------------------------------------------------------   */ @@ -1541,9 +1588,9 @@ TestdelCmd(  	return TCL_ERROR;      } -    dPtr = (DelCmd *) ckalloc(sizeof(DelCmd)); +    dPtr = ckalloc(sizeof(DelCmd));      dPtr->interp = interp; -    dPtr->deleteCmd = (char *) ckalloc((unsigned) (strlen(argv[3]) + 1)); +    dPtr->deleteCmd = ckalloc(strlen(argv[3]) + 1);      strcpy(dPtr->deleteCmd, argv[3]);      Tcl_CreateCommand(slave, argv[2], DelCmdProc, (ClientData) dPtr, @@ -1562,7 +1609,7 @@ DelCmdProc(      Tcl_AppendResult(interp, dPtr->deleteCmd, NULL);      ckfree(dPtr->deleteCmd); -    ckfree((char *) dPtr); +    ckfree(dPtr);      return TCL_OK;  } @@ -1570,12 +1617,12 @@ static void  DelDeleteProc(      ClientData clientData)	/* String command to evaluate. */  { -    DelCmd *dPtr = (DelCmd *) clientData; +    DelCmd *dPtr = clientData;      Tcl_Eval(dPtr->interp, dPtr->deleteCmd);      Tcl_ResetResult(dPtr->interp);      ckfree(dPtr->deleteCmd); -    ckfree((char *) dPtr); +    ckfree(dPtr);  }  /* @@ -1613,6 +1660,102 @@ TestdelassocdataCmd(  }  /* + *----------------------------------------------------------------------------- + * + * TestdoubledigitsCmd -- + * + *	This procedure implements the 'testdoubledigits' command. It is + *	used to test the low-level floating-point formatting primitives + *	in Tcl. + * + * Usage: + *	testdoubledigits fpval ndigits type ?shorten" + * + * Parameters: + *	fpval - Floating-point value to format. + *	ndigits - Digit count to request from Tcl_DoubleDigits + *	type - One of 'shortest', 'Steele', 'e', 'f' + *	shorten - Indicates that the 'shorten' flag should be passed in. + * + *----------------------------------------------------------------------------- + */ + +static int +TestdoubledigitsObjCmd(ClientData unused, +				/* NULL */ +		       Tcl_Interp* interp, +				/* Tcl interpreter */ +		       int objc, +				/* Parameter count */ +		       Tcl_Obj* const objv[]) +				/* Parameter vector */ +{ +    static const char* options[] = { +	"shortest", +	"Steele", +	"e", +	"f", +	NULL +    }; +    static const int types[] = { +	TCL_DD_SHORTEST, +	TCL_DD_STEELE, +	TCL_DD_E_FORMAT, +	TCL_DD_F_FORMAT +    }; + +    const Tcl_ObjType* doubleType; +    double d; +    int status; +    int ndigits; +    int type; +    int decpt; +    int signum; +    char* str; +    char* endPtr; +    Tcl_Obj* strObj; +    Tcl_Obj* retval; + +    if (objc < 4 || objc > 5) { +	Tcl_WrongNumArgs(interp, 1, objv, "fpval ndigits type ?shorten?"); +	return TCL_ERROR; +    } +    status = Tcl_GetDoubleFromObj(interp, objv[1], &d); +    if (status != TCL_OK) { +	doubleType = Tcl_GetObjType("double"); +	if (objv[1]->typePtr == doubleType +	    || TclIsNaN(objv[1]->internalRep.doubleValue)) { +	    status = TCL_OK; +	    memcpy(&d, &(objv[1]->internalRep.doubleValue), sizeof(double)); +	} +    } +    if (status != TCL_OK +	|| Tcl_GetIntFromObj(interp, objv[2], &ndigits) != TCL_OK +	|| Tcl_GetIndexFromObj(interp, objv[3], options, "conversion type", +			       TCL_EXACT, &type) != TCL_OK) { +	fprintf(stderr, "bad value? %g\n", d); +	return TCL_ERROR; +    } +    type = types[type]; +    if (objc > 4) { +	if (strcmp(Tcl_GetString(objv[4]), "shorten")) { +	    Tcl_SetObjResult(interp, Tcl_NewStringObj("bad flag", -1)); +	    return TCL_ERROR; +	} +	type |= TCL_DD_SHORTEN_FLAG; +    } +    str = TclDoubleDigits(d, ndigits, type, &decpt, &signum, &endPtr); +    strObj = Tcl_NewStringObj(str, endPtr-str); +    ckfree(str); +    retval = Tcl_NewListObj(1, &strObj); +    Tcl_ListObjAppendElement(NULL, retval, Tcl_NewIntObj(decpt)); +    strObj = Tcl_NewStringObj(signum ? "-" : "+", 1); +    Tcl_ListObjAppendElement(NULL, retval, strObj); +    Tcl_SetObjResult(interp, retval); +    return TCL_OK; +} + +/*   *----------------------------------------------------------------------   *   * TestdstringCmd -- @@ -1681,11 +1824,11 @@ TestdstringCmd(  	} else if (strcmp(argv[2], "staticlarge") == 0) {  	    Tcl_SetResult(interp, "first0 first1 first2 first3 first4 first5 first6 first7 first8 first9\nsecond0 second1 second2 second3 second4 second5 second6 second7 second8 second9\nthird0 third1 third2 third3 third4 third5 third6 third7 third8 third9\nfourth0 fourth1 fourth2 fourth3 fourth4 fourth5 fourth6 fourth7 fourth8 fourth9\nfifth0 fifth1 fifth2 fifth3 fifth4 fifth5 fifth6 fifth7 fifth8 fifth9\nsixth0 sixth1 sixth2 sixth3 sixth4 sixth5 sixth6 sixth7 sixth8 sixth9\nseventh0 seventh1 seventh2 seventh3 seventh4 seventh5 seventh6 seventh7 seventh8 seventh9\n", TCL_STATIC);  	} else if (strcmp(argv[2], "free") == 0) { -	    char *s = (char *) ckalloc(100); +	    char *s = ckalloc(100);  	    strcpy(s, "This is a malloc-ed string");  	    Tcl_SetResult(interp, s, TCL_DYNAMIC);  	} else if (strcmp(argv[2], "special") == 0) { -	    char *s = (char *) ckalloc(100) + 16; +	    char *s = (char*)ckalloc(100) + 16;  	    strcpy(s, "This is a specially-allocated string");  	    Tcl_SetResult(interp, s, SpecialFree);  	} else { @@ -1713,7 +1856,7 @@ TestdstringCmd(  	if (Tcl_GetInt(interp, argv[2], &count) != TCL_OK) {  	    return TCL_ERROR;  	} -	Tcl_DStringTrunc(&dstring, count); +	Tcl_DStringSetLength(&dstring, count);      } else if (strcmp(argv[1], "start") == 0) {  	if (argc != 2) {  	    goto wrongNumArgs; @@ -1787,15 +1930,15 @@ TestencodingObjCmd(  	if (objc != 5) {  	    return TCL_ERROR;  	} -	encodingPtr = (TclEncoding *) ckalloc(sizeof(TclEncoding)); +	encodingPtr = ckalloc(sizeof(TclEncoding));  	encodingPtr->interp = interp;  	string = Tcl_GetStringFromObj(objv[3], &length); -	encodingPtr->toUtfCmd = (char *) ckalloc((unsigned) (length + 1)); +	encodingPtr->toUtfCmd = ckalloc(length + 1);  	memcpy(encodingPtr->toUtfCmd, string, (unsigned) length + 1);  	string = Tcl_GetStringFromObj(objv[4], &length); -	encodingPtr->fromUtfCmd = (char *) ckalloc((unsigned) (length + 1)); +	encodingPtr->fromUtfCmd = ckalloc(length + 1);  	memcpy(encodingPtr->fromUtfCmd, string, (unsigned) (length + 1));  	string = Tcl_GetStringFromObj(objv[2], &length); @@ -1890,12 +2033,11 @@ static void  EncodingFreeProc(      ClientData clientData)	/* ClientData associated with type. */  { -    TclEncoding *encodingPtr; +    TclEncoding *encodingPtr = clientData; -    encodingPtr = (TclEncoding *) clientData; -    ckfree((char *) encodingPtr->toUtfCmd); -    ckfree((char *) encodingPtr->fromUtfCmd); -    ckfree((char *) encodingPtr); +    ckfree(encodingPtr->toUtfCmd); +    ckfree(encodingPtr->fromUtfCmd); +    ckfree(encodingPtr);  }  /* @@ -2050,7 +2192,7 @@ TesteventObjCmd(  		"position specifier", TCL_EXACT, &posIndex) != TCL_OK) {  	    return TCL_ERROR;  	} -	ev = (TestEvent *) ckalloc(sizeof(TestEvent)); +	ev = ckalloc(sizeof(TestEvent));  	ev->header.proc = TesteventProc;  	ev->header.nextPtr = NULL;  	ev->interp = interp; @@ -2908,7 +3050,7 @@ TestlinkCmd(  	    if (strcmp(argv[5], "-") == 0) {  		stringVar = NULL;  	    } else { -		stringVar = (char *) ckalloc((unsigned) (strlen(argv[5]) + 1)); +		stringVar = ckalloc(strlen(argv[5]) + 1);  		strcpy(stringVar, argv[5]);  	    }  	} @@ -3015,7 +3157,7 @@ TestlinkCmd(  	    if (strcmp(argv[5], "-") == 0) {  		stringVar = NULL;  	    } else { -		stringVar = (char *) ckalloc((unsigned) (strlen(argv[5]) + 1)); +		stringVar = ckalloc(strlen(argv[5]) + 1);  		strcpy(stringVar, argv[5]);  	    }  	    Tcl_UpdateLinkedVar(interp, "string"); @@ -3136,7 +3278,7 @@ TestlocaleCmd(  	"ctype", "numeric", "time", "collate", "monetary",  	"all",	NULL      }; -    static int lcTypes[] = { +    static const int lcTypes[] = {  	LC_CTYPE, LC_NUMERIC, LC_TIME, LC_COLLATE, LC_MONETARY,  	LC_ALL      }; @@ -3327,7 +3469,7 @@ CleanupTestSetassocdataTests(      ClientData clientData,	/* Data to be released. */      Tcl_Interp *interp)		/* Interpreter being deleted. */  { -    ckfree((char *) clientData); +    ckfree(clientData);  }  /* @@ -3858,10 +4000,8 @@ TestregexpObjCmd(  			info.matches[ii].end - 1);  	    }  	} -	valuePtr = Tcl_ObjSetVar2(interp, varPtr, NULL, newPtr, 0); +	valuePtr = Tcl_ObjSetVar2(interp, varPtr, NULL, newPtr, TCL_LEAVE_ERR_MSG);  	if (valuePtr == NULL) { -	    Tcl_AppendResult(interp, "couldn't set variable \"", -		    Tcl_GetString(varPtr), "\"", NULL);  	    return TCL_ERROR;  	}      } @@ -4026,7 +4166,7 @@ TestsetassocdataCmd(  	return TCL_ERROR;      } -    buf = ckalloc((unsigned) strlen(argv[2]) + 1); +    buf = ckalloc(strlen(argv[2]) + 1);      strcpy(buf, argv[2]);      /* @@ -4131,8 +4271,8 @@ TeststaticpkgCmd(      if (Tcl_GetInt(interp, argv[3], &loaded) != TCL_OK) {  	return TCL_ERROR;      } -    Tcl_StaticPackage((loaded) ? interp : NULL, argv[1], StaticInitProc, -	    (safe) ? StaticInitProc : NULL); +    tclStubsPtr->tcl_StaticPackage((loaded) ? interp : NULL, argv[1], +	    StaticInitProc, (safe) ? StaticInitProc : NULL);      return TCL_OK;  } @@ -4268,8 +4408,26 @@ TestseterrorcodeCmd(  	Tcl_SetResult(interp, "too many args", TCL_STATIC);  	return TCL_ERROR;      } -    Tcl_SetErrorCode(interp, argv[1], argv[2], argv[3], argv[4], -	    argv[5], NULL); +    switch (argc) { +    case 1: +	Tcl_SetErrorCode(interp, "NONE", NULL); +	break; +    case 2: +	Tcl_SetErrorCode(interp, argv[1], NULL); +	break; +    case 3: +	Tcl_SetErrorCode(interp, argv[1], argv[2], NULL); +	break; +    case 4: +	Tcl_SetErrorCode(interp, argv[1], argv[2], argv[3], NULL); +	break; +    case 5: +	Tcl_SetErrorCode(interp, argv[1], argv[2], argv[3], argv[4], NULL); +	break; +    case 6: +	Tcl_SetErrorCode(interp, argv[1], argv[2], argv[3], argv[4], +		argv[5], NULL); +    }      return TCL_ERROR;  } @@ -4408,53 +4566,12 @@ TestpanicCmd(       */      argString = Tcl_Merge(argc-1, argv+1); -    Tcl_Panic(argString); -    ckfree((char *)argString); +    Tcl_Panic("%s", argString); +    ckfree(argString);      return TCL_OK;  } -/* - *---------------------------------------------------------------------- - * - * TestfinexitObjCmd -- - * - *	Calls a variant of [exit] including the full finalization path. - * - * Results: - *	Error, or doesn't return. - * - * Side effects: - *	Exits application. - * - *---------------------------------------------------------------------- - */ - -static int -TestfinexitObjCmd( -    ClientData dummy,		/* Not used. */ -    Tcl_Interp *interp,		/* Current interpreter. */ -    int objc,			/* Number of arguments. */ -    Tcl_Obj *const objv[])	/* Argument objects. */ -{ -    int value; - -    if ((objc != 1) && (objc != 2)) { -	Tcl_WrongNumArgs(interp, 1, objv, "?returnCode?"); -	return TCL_ERROR; -    } - -    if (objc == 1) { -	value = 0; -    } else if (Tcl_GetIntFromObj(interp, objv[1], &value) != TCL_OK) { -	return TCL_ERROR; -    } -    Tcl_Finalize(); -    TclpExit(value); -    /*NOTREACHED*/ -    return TCL_ERROR;		/* Better not ever reach this! */ -} -  static int  TestfileCmd(      ClientData dummy,		/* Not used. */ @@ -4634,8 +4751,8 @@ GetTimesCmd(      fprintf(stderr, "alloc & free 100000 6 word items\n");      Tcl_GetTime(&start);      for (i = 0;  i < 100000;  i++) { -	objPtr = (Tcl_Obj *) ckalloc(sizeof(Tcl_Obj)); -	ckfree((char *) objPtr); +	objPtr = ckalloc(sizeof(Tcl_Obj)); +	ckfree(objPtr);      }      Tcl_GetTime(&stop);      timePer = (stop.sec - start.sec)*1000000 + (stop.usec - start.usec); @@ -4643,10 +4760,10 @@ GetTimesCmd(      /* alloc 5000 times */      fprintf(stderr, "alloc 5000 6 word items\n"); -    objv = (Tcl_Obj **) ckalloc(5000 * sizeof(Tcl_Obj *)); +    objv = ckalloc(5000 * sizeof(Tcl_Obj *));      Tcl_GetTime(&start);      for (i = 0;  i < 5000;  i++) { -	objv[i] = (Tcl_Obj *) ckalloc(sizeof(Tcl_Obj)); +	objv[i] = ckalloc(sizeof(Tcl_Obj));      }      Tcl_GetTime(&stop);      timePer = (stop.sec - start.sec)*1000000 + (stop.usec - start.usec); @@ -4656,7 +4773,7 @@ GetTimesCmd(      fprintf(stderr, "free 5000 6 word items\n");      Tcl_GetTime(&start);      for (i = 0;  i < 5000;  i++) { -	ckfree((char *) objv[i]); +	ckfree(objv[i]);      }      Tcl_GetTime(&stop);      timePer = (stop.sec - start.sec)*1000000 + (stop.usec - start.usec); @@ -4682,7 +4799,7 @@ GetTimesCmd(      Tcl_GetTime(&stop);      timePer = (stop.sec - start.sec)*1000000 + (stop.usec - start.usec);      fprintf(stderr, "   %.3f usec per Tcl_DecrRefCount\n", timePer/5000); -    ckfree((char *) objv); +    ckfree(objv);      /* TclGetString 100000 times */      fprintf(stderr, "TclGetStringFromObj of \"12345\" 100000 times\n"); @@ -4914,6 +5031,7 @@ Testset2Cmd(      }  } +#ifndef TCL_NO_DEPRECATED  /*   *----------------------------------------------------------------------   * @@ -5047,6 +5165,7 @@ TestsaveresultFree(  {      freeCount++;  } +#endif /* TCL_NO_DEPRECATED */  /*   *---------------------------------------------------------------------- @@ -5073,7 +5192,7 @@ TestmainthreadCmd(      const char **argv)		/* Argument strings. */  {      if (argc == 1) { -	Tcl_Obj *idObj = Tcl_NewLongObj((long) Tcl_GetCurrentThread()); +	Tcl_Obj *idObj = Tcl_NewLongObj((long)(size_t)Tcl_GetCurrentThread());  	Tcl_SetObjResult(interp, idObj);  	return TCL_OK; @@ -5230,7 +5349,7 @@ TestChannelCmd(  		    *nextPtrPtr = curPtr->nextPtr;  		    curPtr->nextPtr = NULL;  		    chan = curPtr->chan; -		    ckfree((char *) curPtr); +		    ckfree(curPtr);  		    break;  		}  	    } @@ -5300,7 +5419,7 @@ TestChannelCmd(  	/* Remember the channel in the pool of detached channels */ -	det = (TestChannel *) ckalloc(sizeof(TestChannel)); +	det = ckalloc(sizeof(TestChannel));  	det->chan     = chan;  	det->nextPtr  = firstDetached;  	firstDetached = det; @@ -5470,7 +5589,7 @@ TestChannelCmd(  	    return TCL_ERROR;  	} -	TclFormatInt(buf, (long) Tcl_GetChannelThread(chan)); +	TclFormatInt(buf, (size_t) Tcl_GetChannelThread(chan));  	Tcl_AppendResult(interp, buf, NULL);  	return TCL_OK;      } @@ -5698,8 +5817,7 @@ TestChannelEventCmd(  	    return TCL_ERROR;  	} -	esPtr = (EventScriptRecord *) ckalloc((unsigned) -		sizeof(EventScriptRecord)); +	esPtr = ckalloc(sizeof(EventScriptRecord));  	esPtr->nextPtr = statePtr->scriptRecordPtr;  	statePtr->scriptRecordPtr = esPtr; @@ -5756,7 +5874,7 @@ TestChannelEventCmd(  	Tcl_DeleteChannelHandler((Tcl_Channel) chanPtr,  		TclChannelEventScriptInvoker, (ClientData) esPtr);  	Tcl_DecrRefCount(esPtr->scriptPtr); -	ckfree((char *) esPtr); +	ckfree(esPtr);  	return TCL_OK;      } @@ -5797,7 +5915,7 @@ TestChannelEventCmd(  	    Tcl_DeleteChannelHandler((Tcl_Channel) chanPtr,  		    TclChannelEventScriptInvoker, (ClientData) esPtr);  	    Tcl_DecrRefCount(esPtr->scriptPtr); -	    ckfree((char *) esPtr); +	    ckfree(esPtr);  	}  	statePtr->scriptRecordPtr = NULL;  	return TCL_OK; @@ -6082,7 +6200,7 @@ TestReport(  	 * API, but there you go. We should convert it to objects.  	 */ -	Tcl_SavedResult savedResult; +	Tcl_Obj *savedResult;  	Tcl_DString ds;  	Tcl_DStringInit(&ds); @@ -6096,11 +6214,15 @@ TestReport(  	    Tcl_DStringAppendElement(&ds, Tcl_GetString(arg2));  	}  	Tcl_DStringEndSublist(&ds); -	Tcl_SaveResult(interp, &savedResult); +	savedResult = Tcl_GetObjResult(interp); +	Tcl_IncrRefCount(savedResult); +	Tcl_SetObjResult(interp, Tcl_NewObj());  	Tcl_Eval(interp, Tcl_DStringValue(&ds));  	Tcl_DStringFree(&ds); -	Tcl_RestoreResult(interp, &savedResult); -   } +	Tcl_ResetResult(interp); +	Tcl_SetObjResult(interp, savedResult); +	Tcl_DecrRefCount(savedResult); +    }  }  static int @@ -6525,6 +6647,62 @@ TestNumUtfCharsCmd(      }      return TCL_OK;  } + +#if defined(HAVE_CPUID) || defined(_WIN32) +/* + *---------------------------------------------------------------------- + * + * TestcpuidCmd -- + * + *	Retrieves CPU ID information. + * + * Usage: + *	testwincpuid <eax> + * + * Parameters: + *	eax - The value to pass in the EAX register to a CPUID instruction. + * + * Results: + *	Returns a four-element list containing the values from the EAX, EBX, + *	ECX and EDX registers returned from the CPUID instruction. + * + * Side effects: + *	None. + * + *---------------------------------------------------------------------- + */ + +static int +TestcpuidCmd( +    ClientData dummy, +    Tcl_Interp* interp,		/* Tcl interpreter */ +    int objc,			/* Parameter count */ +    Tcl_Obj *const * objv)	/* Parameter vector */ +{ +    int status, index, i; +    unsigned int regs[4]; +    Tcl_Obj *regsObjs[4]; + +    if (objc != 2) { +	Tcl_WrongNumArgs(interp, 1, objv, "eax"); +	return TCL_ERROR; +    } +    if (Tcl_GetIntFromObj(interp, objv[1], &index) != TCL_OK) { +	return TCL_ERROR; +    } +    status = TclWinCPUID((unsigned) index, regs); +    if (status != TCL_OK) { +	Tcl_SetObjResult(interp, +		Tcl_NewStringObj("operation not available", -1)); +	return status; +    } +    for (i=0 ; i<4 ; ++i) { +	regsObjs[i] = Tcl_NewIntObj((int) regs[i]); +    } +    Tcl_SetObjResult(interp, Tcl_NewListObj(4, regsObjs)); +    return TCL_OK; +} +#endif  /*   * Used to do basic checks of the TCL_HASH_KEY_SYSTEM_HASH flag @@ -6558,7 +6736,7 @@ TestHashSystemHashCmd(      }      for (i=0 ; i<limit ; i++) { -	hPtr = Tcl_CreateHashEntry(&hash, (char *) INT2PTR(i), &isNew); +	hPtr = Tcl_CreateHashEntry(&hash, INT2PTR(i), &isNew);  	if (!isNew) {  	    Tcl_SetObjResult(interp, Tcl_NewIntObj(i));  	    Tcl_AppendToObj(Tcl_GetObjResult(interp)," creation problem",-1); @@ -6642,7 +6820,7 @@ TestNRELevels(      ptrdiff_t depth;      Tcl_Obj *levels[6];      int i = 0; -    TEOV_callback *cbPtr = ((Interp *) interp)->execEnvPtr->callbackPtr; +    NRE_callback *cbPtr = iPtr->execEnvPtr->callbackPtr;      if (refDepth == NULL) {  	refDepth = &depth; @@ -6651,11 +6829,11 @@ TestNRELevels(      depth = (refDepth - &depth);      levels[0] = Tcl_NewIntObj(depth); -    levels[1] = Tcl_NewIntObj(((Interp *)interp)->numLevels); +    levels[1] = Tcl_NewIntObj(iPtr->numLevels);      levels[2] = Tcl_NewIntObj(iPtr->cmdFramePtr->level);      levels[3] = Tcl_NewIntObj(iPtr->varFramePtr->level); -    levels[4] = Tcl_NewIntObj((iPtr->execEnvPtr->execStackPtr->tosPtr -		    - iPtr->execEnvPtr->execStackPtr->stackWords)); +    levels[4] = Tcl_NewIntObj(iPtr->execEnvPtr->execStackPtr->tosPtr +	    - iPtr->execEnvPtr->execStackPtr->stackWords);      while (cbPtr) {  	i++; @@ -6972,9 +7150,245 @@ TestconcatobjCmd(  }  /* + *---------------------------------------------------------------------- + * + * TestparseargsCmd -- + * + *	This procedure implements the "testparseargs" command. It is used to + *	test that Tcl_ParseArgsObjv does indeed return the right number of + *	arguments. In other words, that [Bug 3413857] was fixed properly. + * + * Results: + *	A standard Tcl result. + * + * Side effects: + *	None. + * + *---------------------------------------------------------------------- + */ + +static int +TestparseargsCmd( +    ClientData dummy,		/* Not used. */ +    Tcl_Interp *interp,		/* Current interpreter. */ +    int objc,			/* Number of arguments. */ +    Tcl_Obj *const objv[])	/* Arguments. */ +{ +    static int foo = 0; +    int count = objc; +    Tcl_Obj **remObjv, *result[3]; +    Tcl_ArgvInfo argTable[] = { +        {TCL_ARGV_CONSTANT, "-bool", INT2PTR(1), &foo, "booltest", NULL}, +        TCL_ARGV_AUTO_REST, TCL_ARGV_AUTO_HELP, TCL_ARGV_TABLE_END +    }; + +    foo = 0; +    if (Tcl_ParseArgsObjv(interp, argTable, &count, objv, &remObjv)!=TCL_OK) { +        return TCL_ERROR; +    } +    result[0] = Tcl_NewIntObj(foo); +    result[1] = Tcl_NewIntObj(count); +    result[2] = Tcl_NewListObj(count, remObjv); +    Tcl_SetObjResult(interp, Tcl_NewListObj(3, result)); +    ckfree(remObjv); +    return TCL_OK; +} + +/** + * Test harness for command and variable resolvers. + */ + +static int +InterpCmdResolver( +    Tcl_Interp *interp, +    const char *name, +    Tcl_Namespace *context, +    int flags, +    Tcl_Command *rPtr) +{ +    Interp *iPtr = (Interp *) interp; +    CallFrame *varFramePtr = iPtr->varFramePtr; +    Proc *procPtr = (varFramePtr->isProcCallFrame & FRAME_IS_PROC) ? +            varFramePtr->procPtr : NULL; +    Namespace *ns2NsPtr = (Namespace *) +            Tcl_FindNamespace(interp, "::ns2", NULL, 0); + +    if (procPtr && (procPtr->cmdPtr->nsPtr == iPtr->globalNsPtr +            || (ns2NsPtr && procPtr->cmdPtr->nsPtr == ns2NsPtr))) { +        const char *callingCmdName = +                Tcl_GetCommandName(interp, (Tcl_Command) procPtr->cmdPtr); + +        if ((callingCmdName[0] == 'x') && (callingCmdName[1] == '\0') +                && (name[0] == 'z') && (name[1] == '\0')) { +            Tcl_Command sourceCmdPtr = Tcl_FindCommand(interp, "y", NULL, +                    TCL_GLOBAL_ONLY); + +            if (sourceCmdPtr != NULL) { +                *rPtr = sourceCmdPtr; +                return TCL_OK; +            } +        } +    } +    return TCL_CONTINUE; +} + +static int +InterpVarResolver( +    Tcl_Interp *interp, +    const char *name, +    Tcl_Namespace *context, +    int flags, +    Tcl_Var *rPtr) +{ +    /* +     * Don't resolve the variable; use standard rules. +     */ + +    return TCL_CONTINUE; +} + +typedef struct MyResolvedVarInfo { +    Tcl_ResolvedVarInfo vInfo;  /* This must be the first element. */ +    Tcl_Var var; +    Tcl_Obj *nameObj; +} MyResolvedVarInfo; + +static inline void +HashVarFree( +    Tcl_Var var) +{ +    if (VarHashRefCount(var) < 2) { +        ckfree(var); +    } else { +        VarHashRefCount(var)--; +    } +} + +static void +MyCompiledVarFree( +    Tcl_ResolvedVarInfo *vInfoPtr) +{ +    MyResolvedVarInfo *resVarInfo = (MyResolvedVarInfo *) vInfoPtr; + +    Tcl_DecrRefCount(resVarInfo->nameObj); +    if (resVarInfo->var) { +        HashVarFree(resVarInfo->var); +    } +    ckfree(vInfoPtr); +} + +#define TclVarHashGetValue(hPtr) \ +    ((Var *) ((char *)hPtr - TclOffset(VarInHash, entry))) + +static Tcl_Var +MyCompiledVarFetch( +    Tcl_Interp *interp, +    Tcl_ResolvedVarInfo *vinfoPtr) +{ +    MyResolvedVarInfo *resVarInfo = (MyResolvedVarInfo *) vinfoPtr; +    Tcl_Var var = resVarInfo->var; +    int isNewVar; +    Interp *iPtr = (Interp *) interp; +    Tcl_HashEntry *hPtr; + +    if (var != NULL) { +        if (!(((Var *) var)->flags & VAR_DEAD_HASH)) { +            /* +             * The cached variable is valid, return it. +             */ + +            return var; +        } + +        /* +         * The variable is not valid anymore. Clean it up. +         */ + +        HashVarFree(var); +    } + +    hPtr = Tcl_CreateHashEntry((Tcl_HashTable *) &iPtr->globalNsPtr->varTable, +            (char *) resVarInfo->nameObj, &isNewVar); +    if (hPtr) { +        var = (Tcl_Var) TclVarHashGetValue(hPtr); +    } else { +        var = NULL; +    } +    resVarInfo->var = var; + +    /* +     * Increment the reference counter to avoid ckfree() of the variable in +     * Tcl's FreeVarEntry(); for cleanup, we provide our own HashVarFree(); +     */ + +    VarHashRefCount(var)++; +    return var; +} + +static int +InterpCompiledVarResolver( +    Tcl_Interp *interp, +    const char *name, +    int length, +    Tcl_Namespace *context, +    Tcl_ResolvedVarInfo **rPtr) +{ +    if (*name == 'T') { + 	MyResolvedVarInfo *resVarInfo = ckalloc(sizeof(MyResolvedVarInfo)); + + 	resVarInfo->vInfo.fetchProc = MyCompiledVarFetch; + 	resVarInfo->vInfo.deleteProc = MyCompiledVarFree; + 	resVarInfo->var = NULL; + 	resVarInfo->nameObj = Tcl_NewStringObj(name, -1); + 	Tcl_IncrRefCount(resVarInfo->nameObj); + 	*rPtr = &resVarInfo->vInfo; + 	return TCL_OK; +    } +    return TCL_CONTINUE; +} + +static int +TestInterpResolverCmd( +    ClientData clientData, +    Tcl_Interp *interp, +    int objc, +    Tcl_Obj *const objv[]) +{ +    static const char *const table[] = { +        "down", "up", NULL +    }; +    int idx; +#define RESOLVER_KEY "testInterpResolver" + +    if (objc != 2) { +        Tcl_WrongNumArgs(interp, 1, objv, "up|down"); + 	return TCL_ERROR; +    } +    if (Tcl_GetIndexFromObj(interp, objv[1], table, "operation", TCL_EXACT, +            &idx) != TCL_OK) { +        return TCL_ERROR; +    } +    switch (idx) { +    case 1: /* up */ +        Tcl_AddInterpResolvers(interp, RESOLVER_KEY, InterpCmdResolver, +                InterpVarResolver, InterpCompiledVarResolver); +        break; +    case 0: /*down*/ +        if (!Tcl_RemoveInterpResolvers(interp, RESOLVER_KEY)) { +            Tcl_AppendResult(interp, "could not remove the resolver scheme", +                    NULL); +            return TCL_ERROR; +        } +    } +    return TCL_OK; +} + +/*   * Local Variables:   * mode: c   * c-basic-offset: 4   * fill-column: 78 + * tab-width: 8 + * indent-tabs-mode: nil   * End:   */ | 
