Welcome to mirror list, hosted at ThFree Co, Russian Federation.

cygwin.com/git/newlib-cygwin.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorChristopher Faylor <me@cgf.cx>2002-06-13 05:28:51 +0400
committerChristopher Faylor <me@cgf.cx>2002-06-13 05:28:51 +0400
commit12a2ef446296efe35cc6c433af728ddbc38f8a17 (patch)
tree0d46a9d035c806c861702cfcee96e9cdddb5dc27
parentb89a2aac33badf2aa6e69bb7c35558cb70c5f338 (diff)
* environ.cc (build_env): Correctly fill out windows environment block with
win32 paths rather than posix paths.
-rw-r--r--winsup/cygwin/ChangeLog5
-rw-r--r--winsup/cygwin/environ.cc88
2 files changed, 64 insertions, 29 deletions
diff --git a/winsup/cygwin/ChangeLog b/winsup/cygwin/ChangeLog
index a8d26e36e..0998a992a 100644
--- a/winsup/cygwin/ChangeLog
+++ b/winsup/cygwin/ChangeLog
@@ -1,5 +1,10 @@
2002-06-12 Christopher Faylor <cgf@redhat.com>
+ * environ.cc (build_env): Correctly fill out windows environment block
+ with win32 paths rather than posix paths.
+
+2002-06-12 Christopher Faylor <cgf@redhat.com>
+
* cygheap.cc (cygheap_user::set_name): Set homedrive and homepath to
NULL on user name change.
(cygheap_user::set_logsrv): Allocate enough space for leading \\ so
diff --git a/winsup/cygwin/environ.cc b/winsup/cygwin/environ.cc
index 357af7717..da96b5245 100644
--- a/winsup/cygwin/environ.cc
+++ b/winsup/cygwin/environ.cc
@@ -12,6 +12,7 @@ details. */
#include <stdlib.h>
#include <stddef.h>
#include <ctype.h>
+#include <assert.h>
#include <sys/cygwin.h>
#include <cygwin/version.h>
#include "pinfo.h"
@@ -96,10 +97,9 @@ win_env::add_cache (const char *in_posix, const char *in_native)
/* Check for a "special" environment variable name. *env is the pointer
- * to the beginning of the environment variable name. n is the length
- * of the name including a mandatory '='. Returns a pointer to the
- * appropriate conversion structure.
- */
+ to the beginning of the environment variable name. *in_posix is any
+ known posix value for the environment variable. Returns a pointer to
+ the appropriate conversion structure. */
win_env * __stdcall
getwinenv (const char *env, const char *in_posix)
{
@@ -111,7 +111,7 @@ getwinenv (const char *env, const char *in_posix)
{
win_env * const we = conv_envvars + i;
const char *val;
- if (!cur_environ () || !(val = in_posix ?: getenv(we->name)))
+ if (!cur_environ () || !(val = in_posix ?: getenv (we->name)))
debug_printf ("can't set native for %s since no environ yet",
we->name);
else if (!envcache || !we->posix || strcmp (val, we->posix) != 0)
@@ -824,51 +824,53 @@ char ** __stdcall
build_env (const char * const *envp, char *&envblock, int &envc,
bool no_envblock)
{
- int len, n, tl;
+ int len, n;
const char * const *srcp;
char **dstp;
bool saw_spenv[SPENVS_SIZE] = {0};
debug_printf ("envp %p", envp);
- tl = 0;
-
+ /* How many elements? */
for (n = 0; envp[n]; n++)
continue;
- char **newenv = (char **) cmalloc (HEAP_1_ARGV, sizeof (char *) * (n + SPENVS_SIZE + 1));
+ /* Allocate a new "argv-style" environ list with room for extra stuff. */
+ char **newenv = (char **) cmalloc (HEAP_1_ARGV, sizeof (char *) *
+ (n + SPENVS_SIZE + 1));
+ int tl = 0;
+ /* Iterate over input list, generating a new environment list and refreshing
+ "special" entries, if necessary. */
for (srcp = envp, dstp = newenv; *srcp; srcp++, dstp++)
{
len = strcspn (*srcp, "=") + 1;
+ /* Look for entries that require special attention */
for (unsigned i = 0; i < SPENVS_SIZE; i++)
- if (!saw_spenv[i] && (*dstp = spenvs[i].retrieve (no_envblock,*srcp, len)))
+ if (!saw_spenv[i]
+ && (*dstp = spenvs[i].retrieve (no_envblock, *srcp, len)))
{
saw_spenv[i] = 1;
- goto last;
+ goto next;
}
- win_env *conv;
- if (!(conv = getwinenv (*srcp, *srcp + len)))
- *dstp = cstrdup1 (*srcp);
- else
- *dstp = cstrdup1 (conv->native);
-
- if ((*dstp)[0] == '!' && isdrive ((*dstp) + 1) && (*dstp)[3] == '=')
- **dstp = '=';
+ *dstp = cstrdup1 (*srcp);
- last:
- tl += strlen (*dstp) + 1;
+ next:
+ if (!no_envblock)
+ tl += strlen (*dstp) + 1;
}
+ /* Fill in any required-but-missing environment variables. */
for (unsigned i = 0; i < SPENVS_SIZE; i++)
if (!saw_spenv[i])
{
*dstp = spenvs[i].retrieve (no_envblock);
if (*dstp)
{
- tl += strlen (*dstp) + 1;
+ if (!no_envblock)
+ tl += strlen (*dstp) + 1;
dstp++;
}
}
@@ -886,15 +888,43 @@ build_env (const char * const *envp, char *&envblock, int &envc,
qsort (newenv, envc, sizeof (char *), env_sort);
/* Create an environment block suitable for passing to CreateProcess. */
- char *ptr;
- envblock = (char *) malloc (tl + 2);
- for (srcp = newenv, ptr = envblock; *srcp; srcp++)
+ char *s;
+ envblock = (char *) malloc (2 + tl);
+ int new_tl = 0;
+ for (srcp = newenv, s = envblock; *srcp; srcp++)
{
- len = strlen (*srcp);
- memcpy (ptr, *srcp, len + 1);
- ptr += len + 1;
+ const char *p;
+ win_env *conv;
+ len = strcspn (*srcp, "=") + 1;
+
+ /* See if this entry requires posix->win32 conversion. */
+ conv = getwinenv (*srcp, *srcp + len);
+ if (conv)
+ p = conv->native; /* Use win32 path */
+ else
+ p = *srcp; /* Don't worry about it */
+
+ len = strlen (p);
+ new_tl += len + 1; /* Keep running total of block length so far */
+
+ /* See if we need to increase the size of the block. */
+ if (new_tl > tl)
+ envblock = (char *) realloc (envblock, 2 + (tl += len + 100));
+
+ memcpy (s, p, len + 1);
+
+ /* See if environment variable is "special" in a Windows sense.
+ Under NT, the current directories for visited drives are stored
+ as =C:=\bar. Cygwin converts the '=' to '!' for hopefully obvious
+ reasons. We need to convert it back when building the envblock */
+ if (s[0] == '!' && (isdrive (s + 1) || (s[1] == ':' && s[2] == ':'))
+ && s[3] == '=')
+ *s = '=';
+ s += len + 1;
}
- *ptr = '\0'; /* Two null bytes at the end */
+ *s = '\0'; /* Two null bytes at the end */
+ assert ((s - envblock) <= tl); /* Detect if we somehow ran over end
+ of buffer */
}
return newenv;