From 73dea7905cb5fea2540256bacdfde33fe0ad1a0b Mon Sep 17 00:00:00 2001 From: Michael Meissner Date: Wed, 22 Nov 2000 18:26:10 +0000 Subject: [PATCH] Only reference environ indirectly through a pointer --- newlib/ChangeLog | 18 ++++++++++++++++++ newlib/libc/posix/execl.c | 7 ++++++- newlib/libc/posix/execv.c | 7 ++++++- newlib/libc/stdlib/getenv.c | 6 ++---- newlib/libc/stdlib/getenv_r.c | 13 +++++++++---- newlib/libc/stdlib/setenv_r.c | 29 +++++++++++++++++------------ newlib/libc/stdlib/system.c | 12 +++++++++--- 7 files changed, 67 insertions(+), 25 deletions(-) diff --git a/newlib/ChangeLog b/newlib/ChangeLog index 95d1704b7..c824d1290 100644 --- a/newlib/ChangeLog +++ b/newlib/ChangeLog @@ -1,3 +1,21 @@ +2000-11-22 Michael Meissner + + * libc/posix/execl.c (execl): Don't reference environ directly, + reference it only via a static pointer to avoid problems with some + shared library systems and with different uses of small data where + the user specifies his own version of environ. + * libc/posix/execv.c (execv): Ditto. + * libc/stdlib/getenv_r.c (_findenv_r): Ditto. + * libc/stdlib/setenv_r.c (_setenv_r,_unsetenv_r): Ditto. + * libc/stdlib/system.c (system, !cygwin32 case): Ditto. + + * libc/stdlib/getenv.c (environ): Delete unused reference to + environ. + + * libc/stdlib/getenv_r.c: Make initial comment friendlier to emacs + colorization. + * libc/stdlib/system.c: Ditto. + Tue Nov 21 20:32:21 2000 Christopher Faylor * libc/sys/cygwin/sys/dirent.h: Change definition to avoid necessity of diff --git a/newlib/libc/posix/execl.c b/newlib/libc/posix/execl.c index 075df8eba..098005666 100644 --- a/newlib/libc/posix/execl.c +++ b/newlib/libc/posix/execl.c @@ -6,6 +6,11 @@ #include <_ansi.h> #include +/* Only deal with a pointer to environ, to work around subtle bugs with shared + libraries and/or small data systems where the user declares his own + 'environ'. */ +static char ***p_environ = &environ; + #ifdef _HAVE_STDC #include @@ -38,5 +43,5 @@ execl (path, arg0, va_alist) while (argv[i++] != NULL); va_end (args); - return _execve (path, (char * _CONST *) argv, environ); + return _execve (path, (char * _CONST *) argv, *p_environ); } diff --git a/newlib/libc/posix/execv.c b/newlib/libc/posix/execv.c index 42a78cb52..5effb9c57 100644 --- a/newlib/libc/posix/execv.c +++ b/newlib/libc/posix/execv.c @@ -6,10 +6,15 @@ #include <_ansi.h> #include +/* Only deal with a pointer to environ, to work around subtle bugs with shared + libraries and/or small data systems where the user declares his own + 'environ'. */ +static char ***p_environ = &environ; + int _DEFUN (execv, (path, argv), const char *path _AND char * const argv[]) { - return _execve (path, (char * _CONST *) argv, environ); + return _execve (path, (char * _CONST *) argv, *p_environ); } diff --git a/newlib/libc/stdlib/getenv.c b/newlib/libc/stdlib/getenv.c index c9f1ecd96..267f67aac 100644 --- a/newlib/libc/stdlib/getenv.c +++ b/newlib/libc/stdlib/getenv.c @@ -18,7 +18,7 @@ TRAD_SYNOPSIS DESCRIPTION <> searches the list of environment variable names and values -(using the global pointer `<>') for a variable whose +(using the global pointer ``<>'') for a variable whose name matches the string at <[name]>. If a variable name matches, <> returns a pointer to the associated value. @@ -39,7 +39,7 @@ variables vary from one system to another. */ /* - * Copyright (c) 1987 Regents of the University of California. + * Copyright (c) 1987, 2000 Regents of the University of California. * All rights reserved. * * Redistribution and use in source and binary forms are permitted @@ -63,8 +63,6 @@ variables vary from one system to another. #include #include -extern char **environ; - /* * _findenv -- * Returns pointer to value associated with name, if any, else NULL. diff --git a/newlib/libc/stdlib/getenv_r.c b/newlib/libc/stdlib/getenv_r.c index 2ccb89a17..dbfcdf54b 100644 --- a/newlib/libc/stdlib/getenv_r.c +++ b/newlib/libc/stdlib/getenv_r.c @@ -19,7 +19,7 @@ TRAD_SYNOPSIS DESCRIPTION <<_getenv_r>> searches the list of environment variable names and values -(using the global pointer `<>') for a variable whose +(using the global pointer ``<>'') for a variable whose name matches the string at <[name]>. If a variable name matches, <<_getenv_r>> returns a pointer to the associated value. @@ -65,6 +65,11 @@ variables vary from one system to another. extern char **environ; +/* Only deal with a pointer to environ, to work around subtle bugs with shared + libraries and/or small data systems where the user declares his own + 'environ'. */ +static char ***p_environ = &environ; + /* * _findenv -- * Returns pointer to value associated with name, if any, else NULL. @@ -89,7 +94,7 @@ _DEFUN (_findenv_r, (reent_ptr, name, offset), /* In some embedded systems, this does not get set. This protects newlib from dereferencing a bad pointer. */ - if (!environ) + if (!*p_environ) return NULL; c = name; @@ -100,11 +105,11 @@ _DEFUN (_findenv_r, (reent_ptr, name, offset), len++; } - for (p = environ; *p; ++p) + for (p = *p_environ; *p; ++p) if (!strncmp (*p, name, len)) if (*(c = *p + len) == '=') { - *offset = p - environ; + *offset = p - *p_environ; ENV_UNLOCK; return (char *) (++c); } diff --git a/newlib/libc/stdlib/setenv_r.c b/newlib/libc/stdlib/setenv_r.c index c012e22a3..2fe34e275 100644 --- a/newlib/libc/stdlib/setenv_r.c +++ b/newlib/libc/stdlib/setenv_r.c @@ -29,6 +29,13 @@ #include #include "envlock.h" +extern char **environ; + +/* Only deal with a pointer to environ, to work around subtle bugs with shared + libraries and/or small data systems where the user declares his own + 'environ'. */ +static char ***p_environ = &environ; + /* _findenv_r is defined in getenv_r.c. */ extern char *_findenv_r _PARAMS ((struct _reent *, const char *, int *)); @@ -45,7 +52,6 @@ _DEFUN (_setenv_r, (reent_ptr, name, value, rewrite), _CONST char *value _AND int rewrite) { - extern char **environ; static int alloced; /* if allocated space before */ register char *C; int l_value, offset; @@ -74,12 +80,12 @@ _DEFUN (_setenv_r, (reent_ptr, name, value, rewrite), register int cnt; register char **P; - for (P = environ, cnt = 0; *P; ++P, ++cnt); + for (P = *p_environ, cnt = 0; *P; ++P, ++cnt); if (alloced) { /* just increase size */ - environ = (char **) _realloc_r (reent_ptr, (char *) environ, - (size_t) (sizeof (char *) * (cnt + 2))); - if (!environ) + *p_environ = (char **) _realloc_r (reent_ptr, (char *) environ, + (size_t) (sizeof (char *) * (cnt + 2))); + if (!*p_environ) { ENV_UNLOCK; return -1; @@ -94,20 +100,20 @@ _DEFUN (_setenv_r, (reent_ptr, name, value, rewrite), ENV_UNLOCK; return (-1); } - bcopy ((char *) environ, (char *) P, cnt * sizeof (char *)); - environ = P; + bcopy ((char *) *p_environ, (char *) P, cnt * sizeof (char *)); + *p_environ = P; } - environ[cnt + 1] = NULL; + (*p_environ)[cnt + 1] = NULL; offset = cnt; } for (C = (char *) name; *C && *C != '='; ++C); /* no `=' in name */ - if (!(environ[offset] = /* name + `=' + value */ + if (!((*p_environ)[offset] = /* name + `=' + value */ _malloc_r (reent_ptr, (size_t) ((int) (C - name) + l_value + 2)))) { ENV_UNLOCK; return -1; } - for (C = environ[offset]; (*C = *name++) && *C != '='; ++C); + for (C = (*p_environ)[offset]; (*C = *name++) && *C != '='; ++C); for (*C++ = '='; (*C++ = *value++) != 0;); ENV_UNLOCK; @@ -124,14 +130,13 @@ _DEFUN (_unsetenv_r, (reent_ptr, name), struct _reent *reent_ptr _AND _CONST char *name) { - extern char **environ; register char **P; int offset; ENV_LOCK; while (_findenv_r (reent_ptr, name, &offset)) /* if set multiple times */ - for (P = &environ[offset];; ++P) + for (P = &(*p_environ)[offset];; ++P) if (!(*P = *(P + 1))) break; diff --git a/newlib/libc/stdlib/system.c b/newlib/libc/stdlib/system.c index 6a4a33c0b..770996e57 100644 --- a/newlib/libc/stdlib/system.c +++ b/newlib/libc/stdlib/system.c @@ -27,7 +27,7 @@ DESCRIPTION Use <> to pass a command string <<*<[s]>>> to <> on your system, and wait for it to finish executing. -Use `<>' to test whether your system has <> +Use ``<>'' to test whether your system has <> available. The alternate function <<_system_r>> is a reentrant version. The @@ -107,6 +107,13 @@ system (s) #endif #if defined (unix) && !defined (__CYGWIN__) +extern char **environ; + +/* Only deal with a pointer to environ, to work around subtle bugs with shared + libraries and/or small data systems where the user declares his own + 'environ'. */ +static char ***p_environ = &environ; + static int do_system (ptr, s) struct _reent *ptr; @@ -114,7 +121,6 @@ do_system (ptr, s) { char *argv[4]; int pid, status; - extern char **environ; argv[0] = "sh"; argv[1] = "-c"; @@ -123,7 +129,7 @@ do_system (ptr, s) if ((pid = _fork_r (ptr)) == 0) { - _execve ("/bin/sh", argv, environ); + _execve ("/bin/sh", argv, *p_environ); exit (100); } else if (pid == -1)