|
@@ -1,143 +1,200 @@
|
|
|
-
|
|
|
-This file is part of the GNU C Library.
|
|
|
-
|
|
|
-The GNU C Library is free software; you can redistribute it and/or
|
|
|
-modify it under the terms of the GNU Library General Public License as
|
|
|
-published by the Free Software Foundation; either version 2 of the
|
|
|
-License, or (at your option) any later version.
|
|
|
-
|
|
|
-The GNU C Library is distributed in the hope that it will be useful,
|
|
|
-but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
-Library General Public License for more details.
|
|
|
-
|
|
|
-You should have received a copy of the GNU Library General Public
|
|
|
-License along with the GNU C Library; see the file COPYING.LIB. If
|
|
|
-not, write to the Free Software Foundation, Inc., 675 Mass Ave,
|
|
|
-Cambridge, MA 02139, USA. */
|
|
|
-
|
|
|
+
|
|
|
+ This file is part of the GNU C Library.
|
|
|
+
|
|
|
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
+ modify it under the terms of the GNU Lesser General Public
|
|
|
+ License as published by the Free Software Foundation; either
|
|
|
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
+
|
|
|
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
+ Lesser General Public License for more details.
|
|
|
+
|
|
|
+ You should have received a copy of the GNU Lesser General Public
|
|
|
+ License along with the GNU C Library; if not, write to the Free
|
|
|
+ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
|
|
+ 02111-1307 USA.
|
|
|
+
|
|
|
+ modified for uClibc by Erik Andersen <andersen@codepoet.org>
|
|
|
+ */
|
|
|
+
|
|
|
+#include <features.h>
|
|
|
+#include <errno.h>
|
|
|
#include <stdlib.h>
|
|
|
#include <string.h>
|
|
|
#include <unistd.h>
|
|
|
-#include <errno.h>
|
|
|
-
|
|
|
|
|
|
-#if defined(_REENTRENT) || defined(_THREAD_SAFE)
|
|
|
-# include <pthread.h>
|
|
|
+#ifdef __UCLIBC_HAS_THREADS__
|
|
|
+
|
|
|
+# include <bits/libc-lock.h>
|
|
|
+__libc_lock_define_initialized (static, envlock)
|
|
|
+# define LOCK __libc_lock_lock (envlock)
|
|
|
+# define UNLOCK __libc_lock_unlock (envlock)
|
|
|
+#else
|
|
|
+# define LOCK
|
|
|
+# define UNLOCK
|
|
|
+#endif
|
|
|
+
|
|
|
+
|
|
|
+ environment. */
|
|
|
+static char **last_environ;
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ the two functions is that for the former must create a new string which
|
|
|
+ is then placed in the environment, while the argument of `putenv'
|
|
|
+ must be used directly. This is all complicated by the fact that we try
|
|
|
+ to reuse values once generated for a `setenv' call since we can never
|
|
|
+ free the strings. */
|
|
|
+int __add_to_environ (const char *name, const char *value,
|
|
|
+ const char *combined, int replace)
|
|
|
+{
|
|
|
+ register char **ep;
|
|
|
+ register size_t size;
|
|
|
+ const size_t namelen = strlen (name);
|
|
|
+ const size_t vallen = value != NULL ? strlen (value) + 1 : 0;
|
|
|
+
|
|
|
+ LOCK;
|
|
|
+
|
|
|
+
|
|
|
+ since another thread might have created a new environment. */
|
|
|
+ ep = __environ;
|
|
|
+
|
|
|
+ size = 0;
|
|
|
+ if (ep != NULL) {
|
|
|
+ for (; *ep != NULL; ++ep) {
|
|
|
+ if (!strncmp (*ep, name, namelen) && (*ep)[namelen] == '=')
|
|
|
+ break;
|
|
|
+ else
|
|
|
+ ++size;
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
-
|
|
|
- by pthread function 'pthread_once'. For this we need a once block. */
|
|
|
-static pthread_once__t _once_block = pthread_once_init;
|
|
|
+ if (ep == NULL || *ep == NULL) {
|
|
|
+ char **new_environ;
|
|
|
|
|
|
-
|
|
|
- modifications. */
|
|
|
-static pthread_mutex_t _setenv_mutex;
|
|
|
+
|
|
|
+ new_environ = (char **) realloc (last_environ,
|
|
|
+ (size + 2) * sizeof (char *));
|
|
|
+ if (new_environ == NULL) {
|
|
|
+ UNLOCK;
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
|
|
|
-static void
|
|
|
-DEFUN_VOID(_init_setenv_mutex)
|
|
|
-{
|
|
|
- pthread_mutex_init(&_setenv_mutex, pthread_mutexattr_default);
|
|
|
-}
|
|
|
+
|
|
|
+ if (combined != NULL) {
|
|
|
+
|
|
|
+ to the user. */
|
|
|
+ new_environ[size] = (char *) combined;
|
|
|
+ } else {
|
|
|
+
|
|
|
+ new_environ[size] = (char *) malloc (namelen + 1 + vallen);
|
|
|
+ if (new_environ[size] == NULL) {
|
|
|
+ __set_errno (ENOMEM);
|
|
|
+ UNLOCK;
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
|
|
|
-# define LOCK() \
|
|
|
- do { pthread_once(&_once_block, _init_setenv_mutex);
|
|
|
- pthread_mutex_lock(&_setenv_mutex); } while (0)
|
|
|
-# define UNLOCK() pthread_mutex_unlock(&_setenv_mutex)
|
|
|
+ memcpy (new_environ[size], name, namelen);
|
|
|
+ new_environ[size][namelen] = '=';
|
|
|
+ memcpy (&new_environ[size][namelen + 1], value, vallen);
|
|
|
+ }
|
|
|
|
|
|
-#else
|
|
|
+ if (__environ != last_environ) {
|
|
|
+ memcpy ((char *) new_environ, (char *) __environ,
|
|
|
+ size * sizeof (char *));
|
|
|
+ }
|
|
|
|
|
|
-# define LOCK()
|
|
|
-# define UNLOCK()
|
|
|
+ new_environ[size + 1] = NULL;
|
|
|
+ last_environ = __environ = new_environ;
|
|
|
+ } else if (replace) {
|
|
|
+ char *np;
|
|
|
+
|
|
|
+
|
|
|
+ if (combined != NULL) {
|
|
|
+ np = (char *) combined;
|
|
|
+ } else {
|
|
|
+ np = malloc (namelen + 1 + vallen);
|
|
|
+ if (np == NULL) {
|
|
|
+ UNLOCK;
|
|
|
+ return -1;
|
|
|
+ }
|
|
|
+ memcpy (np, name, namelen);
|
|
|
+ np[namelen] = '=';
|
|
|
+ memcpy (&np[namelen + 1], value, vallen);
|
|
|
+ }
|
|
|
+ *ep = np;
|
|
|
+ }
|
|
|
|
|
|
-#endif
|
|
|
+ UNLOCK;
|
|
|
+ return 0;
|
|
|
+}
|
|
|
|
|
|
-int setenv(const char *name, const char *value, int replace)
|
|
|
+int setenv (const char *name, const char *value, int replace)
|
|
|
{
|
|
|
- register char **ep;
|
|
|
- register size_t size;
|
|
|
- const size_t namelen = strlen (name);
|
|
|
- const size_t vallen = strlen (value);
|
|
|
- int result = 0;
|
|
|
-
|
|
|
- LOCK();
|
|
|
-
|
|
|
- size = 0;
|
|
|
- for (ep = __environ; *ep != NULL; ++ep)
|
|
|
- if (!memcmp (*ep, name, namelen) && (*ep)[namelen] == '=')
|
|
|
- break;
|
|
|
- else
|
|
|
- ++size;
|
|
|
-
|
|
|
- if (*ep == NULL)
|
|
|
- {
|
|
|
- static char **last_environ = NULL;
|
|
|
- char **new_environ = (char **) malloc((size + 2) * sizeof(char *));
|
|
|
- if (new_environ == NULL)
|
|
|
- {
|
|
|
- result = -1;
|
|
|
- goto do_return;
|
|
|
- }
|
|
|
- (void) memcpy((__ptr_t) new_environ, (__ptr_t) __environ, size * sizeof(char *));
|
|
|
-
|
|
|
- new_environ[size] = malloc (namelen + 1 + vallen + 1);
|
|
|
- if (new_environ[size] == NULL)
|
|
|
- {
|
|
|
- free (new_environ);
|
|
|
- __set_errno(ENOMEM);
|
|
|
- result = -1;
|
|
|
- goto do_return;
|
|
|
- }
|
|
|
- memcpy (new_environ[size], name, namelen);
|
|
|
- new_environ[size][namelen] = '=';
|
|
|
- memcpy (&new_environ[size][namelen + 1], value, vallen + 1);
|
|
|
+ return __add_to_environ (name, value, NULL, replace);
|
|
|
+}
|
|
|
|
|
|
- new_environ[size + 1] = NULL;
|
|
|
+int unsetenv (const char *name)
|
|
|
+{
|
|
|
+ size_t len;
|
|
|
+ char **ep;
|
|
|
|
|
|
- if (last_environ != NULL)
|
|
|
- free ((__ptr_t) last_environ);
|
|
|
- last_environ = new_environ;
|
|
|
- __environ = new_environ;
|
|
|
+ if (name == NULL || *name == '\0' || strchr (name, '=') != NULL) {
|
|
|
+ __set_errno (EINVAL);
|
|
|
+ return -1;
|
|
|
}
|
|
|
- else if (replace)
|
|
|
- {
|
|
|
- size_t len = strlen (*ep);
|
|
|
- if (len < namelen + 1 + vallen)
|
|
|
- {
|
|
|
- char *new = malloc (namelen + 1 + vallen + 1);
|
|
|
- if (new == NULL)
|
|
|
- {
|
|
|
- result = -1;
|
|
|
- goto do_return;
|
|
|
- }
|
|
|
- *ep = new;
|
|
|
- memcpy (*ep, name, namelen);
|
|
|
- (*ep)[namelen] = '=';
|
|
|
+
|
|
|
+ len = strlen (name);
|
|
|
+ LOCK;
|
|
|
+ ep = __environ;
|
|
|
+ while (*ep != NULL) {
|
|
|
+ if (!strncmp (*ep, name, len) && (*ep)[len] == '=') {
|
|
|
+
|
|
|
+ char **dp = ep;
|
|
|
+ do {
|
|
|
+ dp[0] = dp[1];
|
|
|
+ } while (*dp++);
|
|
|
+
|
|
|
+ } else {
|
|
|
+ ++ep;
|
|
|
}
|
|
|
- memcpy (&(*ep)[namelen + 1], value, vallen + 1);
|
|
|
}
|
|
|
-
|
|
|
-do_return:
|
|
|
- UNLOCK();
|
|
|
- return result;
|
|
|
+ UNLOCK;
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
+
|
|
|
+ never made it. Nevertheless the POSIX.9 standard (POSIX bindings
|
|
|
+ for Fortran 77) requires this function. */
|
|
|
+int clearenv (void)
|
|
|
+{
|
|
|
+ LOCK;
|
|
|
+ if (__environ == last_environ && __environ != NULL) {
|
|
|
+
|
|
|
+ free (__environ);
|
|
|
+ last_environ = NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ __environ = NULL;
|
|
|
+ UNLOCK;
|
|
|
+ return 0;
|
|
|
+}
|
|
|
|
|
|
-int unsetenv (const char *name)
|
|
|
+
|
|
|
+int putenv (char *string)
|
|
|
{
|
|
|
- register char **ep;
|
|
|
- register char **dp;
|
|
|
- const size_t namelen = strlen (name);
|
|
|
-
|
|
|
- LOCK();
|
|
|
-
|
|
|
- for (dp = ep = __environ; *ep != NULL; ++ep)
|
|
|
- if (memcmp (*ep, name, namelen) || (*ep)[namelen] != '=')
|
|
|
- {
|
|
|
- *dp = *ep;
|
|
|
- ++dp;
|
|
|
- }
|
|
|
- *dp = NULL;
|
|
|
-
|
|
|
- UNLOCK();
|
|
|
- return 0;
|
|
|
+ int result;
|
|
|
+ const char *const name_end = strchr (string, '=');
|
|
|
+
|
|
|
+ if (name_end != NULL) {
|
|
|
+ char *name = strndup(string, name_end - string);
|
|
|
+ result = __add_to_environ (name, NULL, string, 1);
|
|
|
+ free(name);
|
|
|
+ return(result);
|
|
|
+ }
|
|
|
+ __unsetenv (string);
|
|
|
+ return 0;
|
|
|
}
|
|
|
+
|