diff options
Diffstat (limited to 'generic/tclEnv.c')
| -rw-r--r-- | generic/tclEnv.c | 902 | 
1 files changed, 520 insertions, 382 deletions
| diff --git a/generic/tclEnv.c b/generic/tclEnv.c index 8b46bb2..cd1a954 100644 --- a/generic/tclEnv.c +++ b/generic/tclEnv.c @@ -1,164 +1,182 @@ -/*  +/*   * tclEnv.c --   * - *	Tcl support for environment variables, including a setenv - *	procedure.  This file contains the generic portion of the - *	environment module.  It is primarily responsible for keeping - *	the "env" arrays in sync with the system environment variables. + *	Tcl support for environment variables, including a setenv function. + *	This file contains the generic portion of the environment module. It + *	is primarily responsible for keeping the "env" arrays in sync with the + *	system environment variables.   *   * Copyright (c) 1991-1994 The Regents of the University of California. - * Copyright (c) 1994-1996 Sun Microsystems, Inc. - * - * See the file "license.terms" for information on usage and redistribution - * of this file, and for a DISCLAIMER OF ALL WARRANTIES. + * Copyright (c) 1994-1998 Sun Microsystems, Inc.   * - * SCCS: @(#) tclEnv.c 1.54 97/10/27 17:47:52 + * See the file "license.terms" for information on usage and redistribution of + * this file, and for a DISCLAIMER OF ALL WARRANTIES.   */  #include "tclInt.h" -#include "tclPort.h" -/* - * The structure below is used to keep track of all of the interpereters - * for which we're managing the "env" array.  It's needed so that they - * can all be updated whenever an environment variable is changed - * anywhere. - */ +TCL_DECLARE_MUTEX(envMutex)	/* To serialize access to environ. */ -typedef struct EnvInterp { -    Tcl_Interp *interp;		/* Interpreter for which we're managing -				 * the env array. */ -    struct EnvInterp *nextPtr;	/* Next in list of all such interpreters, -				 * or zero. */ -} EnvInterp; - -static EnvInterp *firstInterpPtr = NULL; -				/* First in list of all managed interpreters, -				 * or NULL if none. */ - -static int cacheSize = 0;	/* Number of env strings in environCache. */ -static char **environCache = NULL; -				/* Array containing all of the environment +static struct { +    int cacheSize;		/* Number of env strings in cache. */ +    char **cache;		/* Array containing all of the environment  				 * strings that Tcl has allocated. */ -  #ifndef USE_PUTENV -static int environSize = 0;	/* Non-zero means that the environ array was +    char **ourEnviron;		/* Cache of the array that we allocate. We +				 * need to track this in case another +				 * subsystem swaps around the environ array +				 * like we do. */ +    int ourEnvironSize;		/* Non-zero means that the environ array was  				 * malloced and has this many total entries  				 * allocated to it (not all may be in use at -				 * once).  Zero means that the environment +				 * once). Zero means that the environment  				 * array is in its original static state. */  #endif +} env;  /* - * Declarations for local procedures defined in this file: + * Declarations for local functions defined in this file:   */ -static char *		EnvTraceProc _ANSI_ARGS_((ClientData clientData, -			    Tcl_Interp *interp, char *name1, char *name2, -			    int flags)); -static int		FindVariable _ANSI_ARGS_((CONST char *name, -			    int *lengthPtr)); -static void		ReplaceString _ANSI_ARGS_((CONST char *oldStr, -			    char *newStr)); -void			TclSetEnv _ANSI_ARGS_((CONST char *name, -			    CONST char *value)); -void			TclUnsetEnv _ANSI_ARGS_((CONST char *name)); +static char *		EnvTraceProc(ClientData clientData, Tcl_Interp *interp, +			    const char *name1, const char *name2, int flags); +static void		ReplaceString(const char *oldStr, char *newStr); +MODULE_SCOPE void	TclSetEnv(const char *name, const char *value); +MODULE_SCOPE void	TclUnsetEnv(const char *name); + +#if defined(__CYGWIN__) +    static void TclCygwinPutenv(char *string); +#   define putenv TclCygwinPutenv +#endif  /*   *----------------------------------------------------------------------   *   * TclSetupEnv --   * - *	This procedure is invoked for an interpreter to make environment - *	variables accessible from that interpreter via the "env" - *	associative array. + *	This function is invoked for an interpreter to make environment + *	variables accessible from that interpreter via the "env" associative + *	array.   *   * Results:   *	None.   *   * Side effects: - *	The interpreter is added to a list of interpreters managed - *	by us, so that its view of envariables can be kept consistent - *	with the view in other interpreters.  If this is the first - *	call to Tcl_SetupEnv, then additional initialization happens, - *	such as copying the environment to dynamically-allocated space - *	for ease of management. + *	The interpreter is added to a list of interpreters managed by us, so + *	that its view of envariables can be kept consistent with the view in + *	other interpreters. If this is the first call to TclSetupEnv, then + *	additional initialization happens, such as copying the environment to + *	dynamically-allocated space for ease of management.   *   *----------------------------------------------------------------------   */  void -TclSetupEnv(interp) -    Tcl_Interp *interp;		/* Interpreter whose "env" array is to be +TclSetupEnv( +    Tcl_Interp *interp)		/* Interpreter whose "env" array is to be  				 * managed. */  { -    EnvInterp *eiPtr; -    char *p, *p2; -    Tcl_DString ds; -    int i, sz; - -#ifdef MAC_TCL -    if (environ == NULL) { -	environSize = TclMacCreateEnv(); -    } -#endif +    Var *varPtr, *arrayPtr; +    Tcl_Obj *varNamePtr; +    Tcl_DString envString; +    Tcl_HashTable namesHash; +    Tcl_HashEntry *hPtr; +    Tcl_HashSearch search;      /* -     * Next, initialize the DString we are going to use for copying -     * the names of the environment variables. +     * Synchronize the values in the environ array with the contents of the +     * Tcl "env" variable. To do this: +     *    1) Remove the trace that fires when the "env" var is updated. +     *    2) Find the existing contents of the "env", storing in a hash table. +     *    3) Create/update elements for each environ variable, removing +     *	     elements from the hash table as we go. +     *    4) Remove the elements for each remaining entry in the hash table, +     *	     which must have existed before yet have no analog in the environ +     *	     variable. +     *    5) Add a trace that synchronizes the "env" array.       */ -    Tcl_DStringInit(&ds); -     +    Tcl_UntraceVar2(interp, "env", NULL, +	    TCL_GLOBAL_ONLY | TCL_TRACE_WRITES | TCL_TRACE_UNSETS | +	    TCL_TRACE_READS | TCL_TRACE_ARRAY, EnvTraceProc, NULL); +      /* -     * Next, add the interpreter to the list of those that we manage. +     * Find out what elements are currently in the global env array.       */ -    eiPtr = (EnvInterp *) ckalloc(sizeof(EnvInterp)); -    eiPtr->interp = interp; -    eiPtr->nextPtr = firstInterpPtr; -    firstInterpPtr = eiPtr; +    TclNewLiteralStringObj(varNamePtr, "env"); +    Tcl_IncrRefCount(varNamePtr); +    Tcl_InitObjHashTable(&namesHash); +    varPtr = TclObjLookupVarEx(interp, varNamePtr, NULL, TCL_GLOBAL_ONLY, +	    /*msg*/ 0, /*createPart1*/ 0, /*createPart2*/ 0, &arrayPtr); +    TclFindArrayPtrElements(varPtr, &namesHash);      /* -     * Store the environment variable values into the interpreter's -     * "env" array, and arrange for us to be notified on future -     * writes and unsets to that array. +     * Go through the environment array and transfer its values into Tcl. At +     * the same time, remove those elements we add/update from the hash table +     * of existing elements, so that after this part processes, that table +     * will hold just the parts to remove.       */ -    (void) Tcl_UnsetVar2(interp, "env", (char *) NULL, TCL_GLOBAL_ONLY); -    for (i = 0; ; i++) { -	p = environ[i]; -	if (p == NULL) { -	    break; -	} -	for (p2 = p; *p2 != '='; p2++) { -	    if (*p2 == 0) { +    if (environ[0] != NULL) { +	int i; + +	Tcl_MutexLock(&envMutex); +	for (i = 0; environ[i] != NULL; i++) { +	    Tcl_Obj *obj1, *obj2; +	    char *p1, *p2; + +	    p1 = Tcl_ExternalToUtfDString(NULL, environ[i], -1, &envString); +	    p2 = strchr(p1, '='); +	    if (p2 == NULL) {  		/* -		 * This condition doesn't seem like it should ever happen, -		 * but it does seem to happen occasionally under some -		 * versions of Solaris; ignore the entry. +		 * This condition seem to happen occasionally under some +		 * versions of Solaris, or when encoding accidents swallow the +		 * '='; ignore the entry.  		 */ -		goto nextEntry; +		continue;  	    } +	    p2++; +	    p2[-1] = '\0'; +	    obj1 = Tcl_NewStringObj(p1, -1); +	    obj2 = Tcl_NewStringObj(p2, -1); +	    Tcl_DStringFree(&envString); + +	    Tcl_IncrRefCount(obj1); +	    Tcl_IncrRefCount(obj2); +	    Tcl_ObjSetVar2(interp, varNamePtr, obj1, obj2, TCL_GLOBAL_ONLY); +	    hPtr = Tcl_FindHashEntry(&namesHash, obj1); +	    if (hPtr != NULL) { +		Tcl_DeleteHashEntry(hPtr); +	    } +	    Tcl_DecrRefCount(obj1); +	    Tcl_DecrRefCount(obj2);  	} -        sz = p2 - p; -        Tcl_DStringSetLength(&ds, 0); -        Tcl_DStringAppend(&ds, p, sz); -	(void) Tcl_SetVar2(interp, "env", Tcl_DStringValue(&ds), -                p2+1, TCL_GLOBAL_ONLY); -	nextEntry: -	continue; +	Tcl_MutexUnlock(&envMutex); +    } + +    /* +     * Delete those elements that existed in the array but which had no +     * counterparts in the environment array. +     */ + +    for (hPtr=Tcl_FirstHashEntry(&namesHash, &search); hPtr!=NULL; +	    hPtr=Tcl_NextHashEntry(&search)) { +	Tcl_Obj *elemName = Tcl_GetHashValue(hPtr); + +	TclObjUnsetVar2(interp, varNamePtr, elemName, TCL_GLOBAL_ONLY);      } -    Tcl_TraceVar2(interp, "env", (char *) NULL, -	    TCL_GLOBAL_ONLY | TCL_TRACE_WRITES | TCL_TRACE_UNSETS, -	    EnvTraceProc, (ClientData) NULL); +    Tcl_DeleteHashTable(&namesHash); +    Tcl_DecrRefCount(varNamePtr);      /* -     * Finally clean up the DString. +     * Re-establish the trace.       */ -    Tcl_DStringFree(&ds); +    Tcl_TraceVar2(interp, "env", NULL, +	    TCL_GLOBAL_ONLY | TCL_TRACE_WRITES | TCL_TRACE_UNSETS | +	    TCL_TRACE_READS | TCL_TRACE_ARRAY, EnvTraceProc, NULL);  }  /* @@ -166,117 +184,149 @@ TclSetupEnv(interp)   *   * TclSetEnv --   * - *	Set an environment variable, replacing an existing value - *	or creating a new variable if there doesn't exist a variable - *	by the given name.  This procedure is intended to be a - *	stand-in for the  UNIX "setenv" procedure so that applications - *	using that procedure will interface properly to Tcl.  To make - *	it a stand-in, the Makefile must define "TclSetEnv" to "setenv". + *	Set an environment variable, replacing an existing value or creating a + *	new variable if there doesn't exist a variable by the given name. This + *	function is intended to be a stand-in for the UNIX "setenv" function + *	so that applications using that function will interface properly to + *	Tcl. To make it a stand-in, the Makefile must define "TclSetEnv" to + *	"setenv".   *   * Results:   *	None.   *   * Side effects: - *	The environ array gets updated, as do all of the interpreters - *	that we manage. + *	The environ array gets updated.   *   *----------------------------------------------------------------------   */  void -TclSetEnv(name, value) -    CONST char *name;		/* Name of variable whose value is to be -				 * set. */ -    CONST char *value;		/* New value for variable. */ +TclSetEnv( +    const char *name,		/* Name of variable whose value is to be set +				 * (UTF-8). */ +    const char *value)		/* New value for variable (UTF-8). */  { -    int index, length, nameLength; +    Tcl_DString envString; +    unsigned nameLength, valueLength; +    int index, length;      char *p, *oldValue; -    EnvInterp *eiPtr; - -#ifdef MAC_TCL -    if (environ == NULL) { -	environSize = TclMacCreateEnv(); -    } -#endif +    const char *p2;      /* -     * Figure out where the entry is going to go.  If the name doesn't -     * already exist, enlarge the array if necessary to make room.  If -     * the name exists, free its old entry. +     * Figure out where the entry is going to go. If the name doesn't already +     * exist, enlarge the array if necessary to make room. If the name exists, +     * free its old entry.       */ -    index = FindVariable(name, &length); +    Tcl_MutexLock(&envMutex); +    index = TclpFindVariable(name, &length); +      if (index == -1) {  #ifndef USE_PUTENV -	if ((length+2) > environSize) { -	    char **newEnviron; - -	    newEnviron = (char **) ckalloc((unsigned) -		    ((length+5) * sizeof(char *))); -	    memcpy((VOID *) newEnviron, (VOID *) environ, -		    length*sizeof(char *)); -	    if (environSize != 0) { -		ckfree((char *) environ); +	/* +	 * We need to handle the case where the environment may be changed +	 * outside our control. ourEnvironSize is only valid if the current +	 * environment is the one we allocated. [Bug 979640] +	 */ + +	if ((env.ourEnviron != environ) || (length+2 > env.ourEnvironSize)) { +	    char **newEnviron = ckalloc((length + 5) * sizeof(char *)); + +	    memcpy(newEnviron, environ, length * sizeof(char *)); +	    if ((env.ourEnvironSize != 0) && (env.ourEnviron != NULL)) { +		ckfree(env.ourEnviron);  	    } -	    environ = newEnviron; -	    environSize = length+5; +	    environ = env.ourEnviron = newEnviron; +	    env.ourEnvironSize = length + 5;  	}  	index = length; -	environ[index+1] = NULL; -#endif +	environ[index + 1] = NULL; +#endif /* USE_PUTENV */  	oldValue = NULL;  	nameLength = strlen(name);      } else { +	const char *env; +  	/* -	 * Compare the new value to the existing value.  If they're -	 * the same then quit immediately (e.g. don't rewrite the -	 * value or propagate it to other interpreters).  Otherwise, -	 * when there are N interpreters there will be N! propagations -	 * of the same value among the interpreters. +	 * Compare the new value to the existing value. If they're the same +	 * then quit immediately (e.g. don't rewrite the value or propagate it +	 * to other interpreters). Otherwise, when there are N interpreters +	 * there will be N! propagations of the same value among the +	 * interpreters.  	 */ -	if (strcmp(value, environ[index]+length+1) == 0) { +	env = Tcl_ExternalToUtfDString(NULL, environ[index], -1, &envString); +	if (strcmp(value, env + (length + 1)) == 0) { +	    Tcl_DStringFree(&envString); +	    Tcl_MutexUnlock(&envMutex);  	    return;  	} +	Tcl_DStringFree(&envString); +  	oldValue = environ[index]; -	nameLength = length; +	nameLength = (unsigned) length;      } -	      /* -     * Create a new entry. +     * Create a new entry. Build a complete UTF string that contains a +     * "name=value" pattern. Then convert the string to the native encoding, +     * and set the environ array value.       */ -    p = (char *) ckalloc((unsigned) (nameLength + strlen(value) + 2)); -    strcpy(p, name); +    valueLength = strlen(value); +    p = ckalloc(nameLength + valueLength + 2); +    memcpy(p, name, nameLength);      p[nameLength] = '='; -    strcpy(p+nameLength+1, value); +    memcpy(p+nameLength+1, value, valueLength+1); +    p2 = Tcl_UtfToExternalDString(NULL, p, -1, &envString);      /* -     * Update the system environment. +     * Copy the native string to heap memory.       */ +    p = ckrealloc(p, Tcl_DStringLength(&envString) + 1); +    memcpy(p, p2, (unsigned) Tcl_DStringLength(&envString) + 1); +    Tcl_DStringFree(&envString); +  #ifdef USE_PUTENV +    /* +     * Update the system environment. +     */ +      putenv(p); +    index = TclpFindVariable(name, &length);  #else      environ[index] = p; -#endif +#endif /* USE_PUTENV */      /* -     * Replace the old value with the new value in the cache. +     * Watch out for versions of putenv that copy the string (e.g. VC++). In +     * this case we need to free the string immediately. Otherwise update the +     * string in the cache.       */ -    ReplaceString(oldValue, p); - -    /* -     * Update all of the interpreters. -     */ +    if ((index != -1) && (environ[index] == p)) { +	ReplaceString(oldValue, p); +#ifdef HAVE_PUTENV_THAT_COPIES +    } else { +	/* +	 * This putenv() copies instead of taking ownership. +	 */ -    for (eiPtr= firstInterpPtr; eiPtr != NULL; eiPtr = eiPtr->nextPtr) { -	(void) Tcl_SetVar2(eiPtr->interp, "env", (char *) name, -		(char *) value, TCL_GLOBAL_ONLY); +	ckfree(p); +#endif /* HAVE_PUTENV_THAT_COPIES */      } +    Tcl_MutexUnlock(&envMutex); + +    if (!strcmp(name, "HOME")) { +	/* +	 * If the user's home directory has changed, we must invalidate the +	 * filesystem cache, because '~' expansions will now be incorrect. +	 */ + +	Tcl_FSMountsChanged(NULL); +    }  }  /* @@ -284,54 +334,50 @@ TclSetEnv(name, value)   *   * Tcl_PutEnv --   * - *	Set an environment variable.  Similar to setenv except that - *	the information is passed in a single string of the form - *	NAME=value, rather than as separate name strings.  This procedure - *	is intended to be a stand-in for the  UNIX "putenv" procedure - *	so that applications using that procedure will interface - *	properly to Tcl.  To make it a stand-in, the Makefile will - *	define "Tcl_PutEnv" to "putenv". + *	Set an environment variable. Similar to setenv except that the + *	information is passed in a single string of the form NAME=value, + *	rather than as separate name strings. This function is intended to be + *	a stand-in for the UNIX "putenv" function so that applications using + *	that function will interface properly to Tcl. To make it a stand-in, + *	the Makefile will define "Tcl_PutEnv" to "putenv".   *   * Results:   *	None.   *   * Side effects: - *	The environ array gets updated, as do all of the interpreters - *	that we manage. + *	The environ array gets updated, as do all of the interpreters that we + *	manage.   *   *----------------------------------------------------------------------   */  int -Tcl_PutEnv(string) -    CONST char *string;		/* Info about environment variable in the -				 * form NAME=value. */ +Tcl_PutEnv( +    const char *assignment)	/* Info about environment variable in the form +				 * NAME=value. (native) */  { -    int nameLength; -    char *name, *value; +    Tcl_DString nameString; +    const char *name; +    char *value; -    if (string == NULL) { +    if (assignment == NULL) {  	return 0;      }      /* -     * Separate the string into name and value parts, then call -     * TclSetEnv to do all of the real work. +     * First convert the native string to UTF. Then separate the string into +     * name and value parts, and call TclSetEnv to do all of the real work.       */ -    value = strchr(string, '='); -    if (value == NULL) { -	return 0; -    } -    nameLength = value - string; -    if (nameLength == 0) { -	return 0; +    name = Tcl_ExternalToUtfDString(NULL, assignment, -1, &nameString); +    value = strchr(name, '='); + +    if ((value != NULL) && (value != name)) { +	value[0] = '\0'; +	TclSetEnv(name, value+1);      } -    name = (char *) ckalloc((unsigned) nameLength+1); -    memcpy((VOID *) name, (VOID *) string, (size_t) nameLength); -    name[nameLength] = 0; -    TclSetEnv(name, value+1); -    ckfree(name); + +    Tcl_DStringFree(&nameString);      return 0;  } @@ -340,11 +386,10 @@ Tcl_PutEnv(string)   *   * TclUnsetEnv --   * - *	Remove an environment variable, updating the "env" arrays - *	in all interpreters managed by us.  This function is intended - *	to replace the UNIX "unsetenv" function (but to do this the - *	Makefile must be modified to redefine "TclUnsetEnv" to - *	"unsetenv". + *	Remove an environment variable, updating the "env" arrays in all + *	interpreters managed by us. This function is intended to replace the + *	UNIX "unsetenv" function (but to do this the Makefile must be modified + *	to redefine "TclUnsetEnv" to "unsetenv".   *   * Results:   *	None. @@ -356,34 +401,32 @@ Tcl_PutEnv(string)   */  void -TclUnsetEnv(name) -    CONST char *name;			/* Name of variable to remove. */ +TclUnsetEnv( +    const char *name)		/* Name of variable to remove (UTF-8). */  { -    EnvInterp *eiPtr;      char *oldValue; -    int length, index; -#ifdef USE_PUTENV +    int length; +    int index; +#ifdef USE_PUTENV_FOR_UNSET +    Tcl_DString envString;      char *string;  #else      char **envPtr; -#endif +#endif /* USE_PUTENV_FOR_UNSET */ -#ifdef MAC_TCL -    if (environ == NULL) { -	environSize = TclMacCreateEnv(); -    } -#endif - -    index = FindVariable(name, &length); +    Tcl_MutexLock(&envMutex); +    index = TclpFindVariable(name, &length);      /* -     * First make sure that the environment variable exists to avoid -     * doing needless work and to avoid recursion on the unset. +     * First make sure that the environment variable exists to avoid doing +     * needless work and to avoid recursion on the unset.       */ -     +      if (index == -1) { +	Tcl_MutexUnlock(&envMutex);  	return;      } +      /*       * Remember the old value so we can free it if Tcl created the string.       */ @@ -391,52 +434,78 @@ TclUnsetEnv(name)      oldValue = environ[index];      /* -     * Update the system environment.  This must be done before we  -     * update the interpreters or we will recurse. +     * Update the system environment. This must be done before we update the +     * interpreters or we will recurse.       */ -#ifdef USE_PUTENV -    string = ckalloc(length+2); -    memcpy((VOID *) string, (VOID *) name, (size_t) length); +#ifdef USE_PUTENV_FOR_UNSET +    /* +     * For those platforms that support putenv to unset, Linux indicates +     * that no = should be included, and Windows requires it. +     */ + +#if defined(_WIN32) || defined(__CYGWIN__) +    string = ckalloc(length + 2); +    memcpy(string, name, (size_t) length);      string[length] = '=';      string[length+1] = '\0'; -    putenv(string); -    ckfree(string);  #else +    string = ckalloc(length + 1); +    memcpy(string, name, (size_t) length); +    string[length] = '\0'; +#endif /* _WIN32 */ + +    Tcl_UtfToExternalDString(NULL, string, -1, &envString); +    string = ckrealloc(string, Tcl_DStringLength(&envString) + 1); +    memcpy(string, Tcl_DStringValue(&envString), +	    (unsigned) Tcl_DStringLength(&envString)+1); +    Tcl_DStringFree(&envString); + +    putenv(string); + +    /* +     * Watch out for versions of putenv that copy the string (e.g. VC++). In +     * this case we need to free the string immediately. Otherwise update the +     * string in the cache. +     */ + +    if (environ[index] == string) { +	ReplaceString(oldValue, string); +#ifdef HAVE_PUTENV_THAT_COPIES +    } else { +	/* +	 * This putenv() copies instead of taking ownership. +	 */ + +	ckfree(string); +#endif /* HAVE_PUTENV_THAT_COPIES */ +    } +#else /* !USE_PUTENV_FOR_UNSET */      for (envPtr = environ+index+1; ; envPtr++) {  	envPtr[-1] = *envPtr;  	if (*envPtr == NULL) {  	    break;  	}      } -#endif - -    /* -     * Replace the old value in the cache. -     */ -      ReplaceString(oldValue, NULL); +#endif /* USE_PUTENV_FOR_UNSET */ -    /* -     * Update all of the interpreters. -     */ - -    for (eiPtr = firstInterpPtr; eiPtr != NULL; eiPtr = eiPtr->nextPtr) { -	(void) Tcl_UnsetVar2(eiPtr->interp, "env", (char *) name, -		TCL_GLOBAL_ONLY); -    } +    Tcl_MutexUnlock(&envMutex);  }  /* - *---------------------------------------------------------------------- + *---------------------------------------------------------------------------   *   * TclGetEnv --   *   *	Retrieve the value of an environment variable.   *   * Results: - *	Returns a pointer to a static string in the environment, - *	or NULL if the value was not found. + *	The result is a pointer to a string specifying the value of the + *	environment variable, or NULL if that environment variable does not + *	exist. Storage for the result string is allocated in valuePtr; the + *	caller must call Tcl_DStringFree() when the result is no longer + *	needed.   *   * Side effects:   *	None. @@ -444,24 +513,37 @@ TclUnsetEnv(name)   *----------------------------------------------------------------------   */ -char * -TclGetEnv(name) -    CONST char *name;		/* Name of variable to find. */ +const char * +TclGetEnv( +    const char *name,		/* Name of environment variable to find +				 * (UTF-8). */ +    Tcl_DString *valuePtr)	/* Uninitialized or free DString in which the +				 * value of the environment variable is +				 * stored. */  {      int length, index; - -#ifdef MAC_TCL -    if (environ == NULL) { -	environSize = TclMacCreateEnv(); -    } -#endif - -    index = FindVariable(name, &length); -    if ((index != -1) &&  (*(environ[index]+length) == '=')) { -	return environ[index]+length+1; -    } else { -	return NULL; +    const char *result; + +    Tcl_MutexLock(&envMutex); +    index = TclpFindVariable(name, &length); +    result = NULL; +    if (index != -1) { +	Tcl_DString envStr; + +	result = Tcl_ExternalToUtfDString(NULL, environ[index], -1, &envStr); +	result += length; +	if (*result == '=') { +	    result++; +	    Tcl_DStringInit(valuePtr); +	    Tcl_DStringAppend(valuePtr, result, -1); +	    result = Tcl_DStringValue(valuePtr); +	} else { +	    result = NULL; +	} +	Tcl_DStringFree(&envStr);      } +    Tcl_MutexUnlock(&envMutex); +    return result;  }  /* @@ -469,62 +551,46 @@ TclGetEnv(name)   *   * EnvTraceProc --   * - *	This procedure is invoked whenever an environment variable - *	is modified or deleted.  It propagates the change to the - *	"environ" array and to any other interpreters for whom - *	we're managing an "env" array. + *	This function is invoked whenever an environment variable is read, + *	modified or deleted. It propagates the change to the global "environ" + *	array.   *   * Results:   *	Always returns NULL to indicate success.   *   * Side effects: - *	Environment variable changes get propagated.  If the whole - *	"env" array is deleted, then we stop managing things for - *	this interpreter (usually this happens because the whole - *	interpreter is being deleted). + *	Environment variable changes get propagated. If the whole "env" array + *	is deleted, then we stop managing things for this interpreter (usually + *	this happens because the whole interpreter is being deleted).   *   *----------------------------------------------------------------------   */  	/* ARGSUSED */  static char * -EnvTraceProc(clientData, interp, name1, name2, flags) -    ClientData clientData;	/* Not used. */ -    Tcl_Interp *interp;		/* Interpreter whose "env" variable is -				 * being modified. */ -    char *name1;		/* Better be "env". */ -    char *name2;		/* Name of variable being modified, or -				 * NULL if whole array is being deleted. */ -    int flags;			/* Indicates what's happening. */ +EnvTraceProc( +    ClientData clientData,	/* Not used. */ +    Tcl_Interp *interp,		/* Interpreter whose "env" variable is being +				 * modified. */ +    const char *name1,		/* Better be "env". */ +    const char *name2,		/* Name of variable being modified, or NULL if +				 * whole array is being deleted (UTF-8). */ +    int flags)			/* Indicates what's happening. */  {      /* -     * First see if the whole "env" variable is being deleted.  If -     * so, just forget about this interpreter. +     * For array traces, let TclSetupEnv do all the work.       */ -    if (name2 == NULL) { -	register EnvInterp *eiPtr, *prevPtr; +    if (flags & TCL_TRACE_ARRAY) { +	TclSetupEnv(interp); +	return NULL; +    } -	if ((flags & (TCL_TRACE_UNSETS|TCL_TRACE_DESTROYED)) -		!= (TCL_TRACE_UNSETS|TCL_TRACE_DESTROYED)) { -	    panic("EnvTraceProc called with confusing arguments"); -	} -	eiPtr = firstInterpPtr; -	if (eiPtr->interp == interp) { -	    firstInterpPtr = eiPtr->nextPtr; -	} else { -	    for (prevPtr = eiPtr, eiPtr = eiPtr->nextPtr; ; -		    prevPtr = eiPtr, eiPtr = eiPtr->nextPtr) { -		if (eiPtr == NULL) { -		    panic("EnvTraceProc couldn't find interpreter"); -		} -		if (eiPtr->interp == interp) { -		    prevPtr->nextPtr = eiPtr->nextPtr; -		    break; -		} -	    } -	} -	ckfree((char *) eiPtr); +    /* +     * If name2 is NULL, then return and do nothing. +     */ + +    if (name2 == NULL) {  	return NULL;      } @@ -533,9 +599,32 @@ EnvTraceProc(clientData, interp, name1, name2, flags)       */      if (flags & TCL_TRACE_WRITES) { -	TclSetEnv(name2, Tcl_GetVar2(interp, "env", name2, TCL_GLOBAL_ONLY)); +	const char *value; + +	value = Tcl_GetVar2(interp, "env", name2, TCL_GLOBAL_ONLY); +	TclSetEnv(name2, value); +    } + +    /* +     * If a value is being read, call TclGetEnv to do all of the work. +     */ + +    if (flags & TCL_TRACE_READS) { +	Tcl_DString valueString; +	const char *value = TclGetEnv(name2, &valueString); + +	if (value == NULL) { +	    Tcl_UnsetVar2(interp, name1, name2, 0); +	    return NULL; +	} +	Tcl_SetVar2(interp, name1, name2, value, 0); +	Tcl_DStringFree(&valueString);      } +    /* +     * For unset traces, let TclUnsetEnv do all the work. +     */ +      if (flags & TCL_TRACE_UNSETS) {  	TclUnsetEnv(name2);      } @@ -547,9 +636,9 @@ EnvTraceProc(clientData, interp, name1, name2, flags)   *   * ReplaceString --   * - *	Replace one string with another in the environment variable - *	cache.  The cache keeps track of all of the environment - *	variables that Tcl has modified so they can be freed later. + *	Replace one string with another in the environment variable cache. The + *	cache keeps track of all of the environment variables that Tcl has + *	modified so they can be freed later.   *   * Results:   *	None. @@ -561,106 +650,55 @@ EnvTraceProc(clientData, interp, name1, name2, flags)   */  static void -ReplaceString(oldStr, newStr) -    CONST char *oldStr;		/* Old environment string. */ -    char *newStr;		/* New environment string. */ +ReplaceString( +    const char *oldStr,		/* Old environment string. */ +    char *newStr)		/* New environment string. */  {      int i; -    char **newCache;      /* -     * Check to see if the old value was allocated by Tcl.  If so, -     * it needs to be deallocated to avoid memory leaks.  Note that this -     * algorithm is O(n), not O(1).  This will result in n-squared behavior -     * if lots of environment changes are being made. +     * Check to see if the old value was allocated by Tcl. If so, it needs to +     * be deallocated to avoid memory leaks. Note that this algorithm is O(n), +     * not O(1). This will result in n-squared behavior if lots of environment +     * changes are being made.       */ -    for (i = 0; i < cacheSize; i++) { -	if ((environCache[i] == oldStr) || (environCache[i] == NULL)) { +    for (i = 0; i < env.cacheSize; i++) { +	if (env.cache[i]==oldStr || env.cache[i]==NULL) {  	    break;  	}      } -    if (i < cacheSize) { +    if (i < env.cacheSize) {  	/*  	 * Replace or delete the old value.  	 */ -	if (environCache[i]) { -	    ckfree(environCache[i]); +	if (env.cache[i]) { +	    ckfree(env.cache[i]);  	} -	     +  	if (newStr) { -	    environCache[i] = newStr; +	    env.cache[i] = newStr;  	} else { -	    for (; i < cacheSize-1; i++) { -		environCache[i] = environCache[i+1]; +	    for (; i < env.cacheSize-1; i++) { +		env.cache[i] = env.cache[i+1];  	    } -	    environCache[cacheSize-1] = NULL; +	    env.cache[env.cacheSize-1] = NULL;  	} -    } else {	 -        int allocatedSize = (cacheSize + 5) * sizeof(char *); - +    } else {  	/*  	 * We need to grow the cache in order to hold the new string.  	 */ -	newCache = (char **) ckalloc((size_t) allocatedSize); -        (VOID *) memset(newCache, (int) 0, (size_t) allocatedSize); -         -	if (environCache) { -	    memcpy((VOID *) newCache, (VOID *) environCache, -		    (size_t) (cacheSize * sizeof(char*))); -	    ckfree((char *) environCache); -	} -	environCache = newCache; -	environCache[cacheSize] = (char *) newStr; -	environCache[cacheSize+1] = NULL; -	cacheSize += 5; -    } -} - -/* - *---------------------------------------------------------------------- - * - * FindVariable -- - * - *	Locate the entry in environ for a given name. - * - * Results: - *	The return value is the index in environ of an entry with the - *	name "name", or -1 if there is no such entry.   The integer at - *	*lengthPtr is filled in with the length of name (if a matching - *	entry is found) or the length of the environ array (if no matching - *	entry is found). - * - * Side effects: - *	None. - * - *---------------------------------------------------------------------- - */ - -static int -FindVariable(name, lengthPtr) -    CONST char *name;		/* Name of desired environment variable. */ -    int *lengthPtr;		/* Used to return length of name (for -				 * successful searches) or number of non-NULL -				 * entries in environ (for unsuccessful -				 * searches). */ -{ -    int i; -    register CONST char *p1, *p2; +	const int growth = 5; -    for (i = 0, p1 = environ[i]; p1 != NULL; i++, p1 = environ[i]) { -	for (p2 = name; *p2 == *p1; p1++, p2++) { -	    /* NULL loop body. */ -	} -	if ((*p1 == '=') && (*p2 == '\0')) { -	    *lengthPtr = p2-name; -	    return i; -	} +	env.cache = ckrealloc(env.cache, +		(env.cacheSize + growth) * sizeof(char *)); +	env.cache[env.cacheSize] = newStr; +	(void) memset(env.cache+env.cacheSize+1, 0, +		(size_t) (growth-1) * sizeof(char *)); +	env.cacheSize += growth;      } -    *lengthPtr = i; -    return -1;  }  /* @@ -668,9 +706,9 @@ FindVariable(name, lengthPtr)   *   * TclFinalizeEnvironment --   * - *	This function releases any storage allocated by this module - *	that isn't still in use by the global environment.  Any - *	strings that are still in the environment will be leaked. + *	This function releases any storage allocated by this module that isn't + *	still in use by the global environment. Any strings that are still in + *	the environment will be leaked.   *   * Results:   *	None. @@ -682,22 +720,122 @@ FindVariable(name, lengthPtr)   */  void -TclFinalizeEnvironment() +TclFinalizeEnvironment(void)  {      /*       * For now we just deallocate the cache array and none of the environment -     * strings.  This may leak more memory that strictly necessary, since some -     * of the strings may no longer be in the environment.  However, +     * strings. This may leak more memory that strictly necessary, since some +     * of the strings may no longer be in the environment. However,       * determining which ones are ok to delete is n-squared, and is pretty       * unlikely, so we don't bother.       */ -    if (environCache) { -	ckfree((char *) environCache); -	environCache = NULL; -	cacheSize    = 0; +    if (env.cache) { +	ckfree(env.cache); +	env.cache = NULL; +	env.cacheSize = 0;  #ifndef USE_PUTENV -	environSize  = 0; +	env.ourEnvironSize = 0;  #endif      }  } + +#if defined(__CYGWIN__) + +/* + * When using cygwin, when an environment variable changes, we need to synch + * with both the cygwin environment (in case the application C code calls + * fork) and the Windows environment (in case the application TCL code calls + * exec, which calls the Windows CreateProcess function). + */ +DLLIMPORT extern void __stdcall SetEnvironmentVariableA(const char*, const char *); + +static void +TclCygwinPutenv( +    char *str) +{ +    char *name, *value; + +    /* +     * Get the name and value, so that we can change the environment variable +     * for Windows. +     */ + +    name = alloca(strlen(str) + 1); +    strcpy(name, str); +    for (value=name ; *value!='=' && *value!='\0' ; ++value) { +	/* Empty body */ +    } +    if (*value == '\0') { +	/* Can't happen. */ +	return; +    } +    *(value++) = '\0'; +    if (*value == '\0') { +	value = NULL; +    } + +    /* +     * Set the cygwin environment variable. +     */ + +#undef putenv +    if (value == NULL) { +	unsetenv(name); +    } else { +	putenv(str); +    } + +    /* +     * Before changing the environment variable in Windows, if this is PATH, +     * we need to convert the value back to a Windows style path. +     * +     * FIXME: The calling program may know it is running under windows, and +     * may have set the path to a Windows path, or, worse, appended or +     * prepended a Windows path to PATH. +     */ + +    if (strcmp(name, "PATH") != 0) { +	/* +	 * If this is Path, eliminate any PATH variable, to prevent any +	 * confusion. +	 */ + +	if (strcmp(name, "Path") == 0) { +	    SetEnvironmentVariableA("PATH", NULL); +	    unsetenv("PATH"); +	} + +	SetEnvironmentVariableA(name, value); +    } else { +	char *buf; + +	/* +	 * Eliminate any Path variable, to prevent any confusion. +	 */ + +	SetEnvironmentVariableA("Path", NULL); +	unsetenv("Path"); + +	if (value == NULL) { +	    buf = NULL; +	} else { +	    int size; + +	    size = cygwin_conv_path_list(0, value, NULL, 0); +	    buf = alloca(size + 1); +	    cygwin_conv_path_list(0, value, buf, size); +	} + +	SetEnvironmentVariableA(name, buf); +    } +} +#endif /* __CYGWIN__ */ + +/* + * Local Variables: + * mode: c + * c-basic-offset: 4 + * fill-column: 78 + * End: + */ | 
