/* Copyright (C) 1992,95,96,97,98,99,2000,2001 Free Software Foundation, Inc.
|
/* Copyright (C) 1992,95,96,97,98,99,2000,2001 Free Software Foundation, Inc.
|
This file is part of the GNU C Library.
|
This file is part of the GNU C Library.
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
The GNU C Library is free software; you can redistribute it and/or
|
modify it under the terms of the GNU Lesser General Public
|
modify it under the terms of the GNU Lesser General Public
|
License as published by the Free Software Foundation; either
|
License as published by the Free Software Foundation; either
|
version 2.1 of the License, or (at your option) any later version.
|
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,
|
The GNU C Library is distributed in the hope that it will be useful,
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
Lesser General Public License for more details.
|
Lesser General Public License for more details.
|
|
|
You should have received a copy of the GNU Lesser General Public
|
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
|
License along with the GNU C Library; if not, write to the Free
|
Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
|
02111-1307 USA.
|
02111-1307 USA.
|
|
|
modified for uClibc by Erik Andersen <andersen@codepoet.org>
|
modified for uClibc by Erik Andersen <andersen@codepoet.org>
|
*/
|
*/
|
|
|
#define _GNU_SOURCE
|
#define _GNU_SOURCE
|
#include <features.h>
|
#include <features.h>
|
#include <errno.h>
|
#include <errno.h>
|
#include <stdlib.h>
|
#include <stdlib.h>
|
#include <string.h>
|
#include <string.h>
|
#include <unistd.h>
|
#include <unistd.h>
|
|
|
#ifdef __UCLIBC_HAS_THREADS__
|
#ifdef __UCLIBC_HAS_THREADS__
|
#include <pthread.h>
|
#include <pthread.h>
|
static pthread_mutex_t mylock = PTHREAD_MUTEX_INITIALIZER;
|
static pthread_mutex_t mylock = PTHREAD_MUTEX_INITIALIZER;
|
# define LOCK __pthread_mutex_lock(&mylock)
|
# define LOCK __pthread_mutex_lock(&mylock)
|
# define UNLOCK __pthread_mutex_unlock(&mylock);
|
# define UNLOCK __pthread_mutex_unlock(&mylock);
|
#else
|
#else
|
# define LOCK
|
# define LOCK
|
# define UNLOCK
|
# define UNLOCK
|
#endif
|
#endif
|
|
|
|
|
/* If this variable is not a null pointer we allocated the current
|
/* If this variable is not a null pointer we allocated the current
|
environment. */
|
environment. */
|
static char **last_environ;
|
static char **last_environ;
|
|
|
|
|
/* This function is used by `setenv' and `putenv'. The difference between
|
/* This function is used by `setenv' and `putenv'. The difference between
|
the two functions is that for the former must create a new string which
|
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'
|
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
|
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
|
to reuse values once generated for a `setenv' call since we can never
|
free the strings. */
|
free the strings. */
|
int __add_to_environ (const char *name, const char *value,
|
int __add_to_environ (const char *name, const char *value,
|
const char *combined, int replace)
|
const char *combined, int replace)
|
{
|
{
|
register char **ep;
|
register char **ep;
|
register size_t size;
|
register size_t size;
|
const size_t namelen = strlen (name);
|
const size_t namelen = strlen (name);
|
const size_t vallen = value != NULL ? strlen (value) + 1 : 0;
|
const size_t vallen = value != NULL ? strlen (value) + 1 : 0;
|
|
|
LOCK;
|
LOCK;
|
|
|
/* We have to get the pointer now that we have the lock and not earlier
|
/* We have to get the pointer now that we have the lock and not earlier
|
since another thread might have created a new environment. */
|
since another thread might have created a new environment. */
|
ep = __environ;
|
ep = __environ;
|
|
|
size = 0;
|
size = 0;
|
if (ep != NULL) {
|
if (ep != NULL) {
|
for (; *ep != NULL; ++ep) {
|
for (; *ep != NULL; ++ep) {
|
if (!strncmp (*ep, name, namelen) && (*ep)[namelen] == '=')
|
if (!strncmp (*ep, name, namelen) && (*ep)[namelen] == '=')
|
break;
|
break;
|
else
|
else
|
++size;
|
++size;
|
}
|
}
|
}
|
}
|
|
|
if (ep == NULL || *ep == NULL) {
|
if (ep == NULL || *ep == NULL) {
|
char **new_environ;
|
char **new_environ;
|
|
|
/* We allocated this space; we can extend it. */
|
/* We allocated this space; we can extend it. */
|
new_environ = (char **) realloc (last_environ,
|
new_environ = (char **) realloc (last_environ,
|
(size + 2) * sizeof (char *));
|
(size + 2) * sizeof (char *));
|
if (new_environ == NULL) {
|
if (new_environ == NULL) {
|
UNLOCK;
|
UNLOCK;
|
return -1;
|
return -1;
|
}
|
}
|
|
|
/* If the whole entry is given add it. */
|
/* If the whole entry is given add it. */
|
if (combined != NULL) {
|
if (combined != NULL) {
|
/* We must not add the string to the search tree since it belongs
|
/* We must not add the string to the search tree since it belongs
|
to the user. */
|
to the user. */
|
new_environ[size] = (char *) combined;
|
new_environ[size] = (char *) combined;
|
} else {
|
} else {
|
/* See whether the value is already known. */
|
/* See whether the value is already known. */
|
new_environ[size] = (char *) malloc (namelen + 1 + vallen);
|
new_environ[size] = (char *) malloc (namelen + 1 + vallen);
|
if (new_environ[size] == NULL) {
|
if (new_environ[size] == NULL) {
|
__set_errno (ENOMEM);
|
__set_errno (ENOMEM);
|
UNLOCK;
|
UNLOCK;
|
return -1;
|
return -1;
|
}
|
}
|
|
|
memcpy (new_environ[size], name, namelen);
|
memcpy (new_environ[size], name, namelen);
|
new_environ[size][namelen] = '=';
|
new_environ[size][namelen] = '=';
|
memcpy (&new_environ[size][namelen + 1], value, vallen);
|
memcpy (&new_environ[size][namelen + 1], value, vallen);
|
}
|
}
|
|
|
if (__environ != last_environ) {
|
if (__environ != last_environ) {
|
memcpy ((char *) new_environ, (char *) __environ,
|
memcpy ((char *) new_environ, (char *) __environ,
|
size * sizeof (char *));
|
size * sizeof (char *));
|
}
|
}
|
|
|
new_environ[size + 1] = NULL;
|
new_environ[size + 1] = NULL;
|
last_environ = __environ = new_environ;
|
last_environ = __environ = new_environ;
|
} else if (replace) {
|
} else if (replace) {
|
char *np;
|
char *np;
|
|
|
/* Use the user string if given. */
|
/* Use the user string if given. */
|
if (combined != NULL) {
|
if (combined != NULL) {
|
np = (char *) combined;
|
np = (char *) combined;
|
} else {
|
} else {
|
np = malloc (namelen + 1 + vallen);
|
np = malloc (namelen + 1 + vallen);
|
if (np == NULL) {
|
if (np == NULL) {
|
UNLOCK;
|
UNLOCK;
|
return -1;
|
return -1;
|
}
|
}
|
memcpy (np, name, namelen);
|
memcpy (np, name, namelen);
|
np[namelen] = '=';
|
np[namelen] = '=';
|
memcpy (&np[namelen + 1], value, vallen);
|
memcpy (&np[namelen + 1], value, vallen);
|
}
|
}
|
*ep = np;
|
*ep = np;
|
}
|
}
|
|
|
UNLOCK;
|
UNLOCK;
|
return 0;
|
return 0;
|
}
|
}
|
|
|
int setenv (const char *name, const char *value, int replace)
|
int setenv (const char *name, const char *value, int replace)
|
{
|
{
|
return __add_to_environ (name, value, NULL, replace);
|
return __add_to_environ (name, value, NULL, replace);
|
}
|
}
|
|
|
int unsetenv (const char *name)
|
int unsetenv (const char *name)
|
{
|
{
|
size_t len;
|
size_t len;
|
char **ep;
|
char **ep;
|
|
|
if (name == NULL || *name == '\0' || strchr (name, '=') != NULL) {
|
if (name == NULL || *name == '\0' || strchr (name, '=') != NULL) {
|
__set_errno (EINVAL);
|
__set_errno (EINVAL);
|
return -1;
|
return -1;
|
}
|
}
|
|
|
len = strlen (name);
|
len = strlen (name);
|
LOCK;
|
LOCK;
|
ep = __environ;
|
ep = __environ;
|
while (*ep != NULL) {
|
while (*ep != NULL) {
|
if (!strncmp (*ep, name, len) && (*ep)[len] == '=') {
|
if (!strncmp (*ep, name, len) && (*ep)[len] == '=') {
|
/* Found it. Remove this pointer by moving later ones back. */
|
/* Found it. Remove this pointer by moving later ones back. */
|
char **dp = ep;
|
char **dp = ep;
|
do {
|
do {
|
dp[0] = dp[1];
|
dp[0] = dp[1];
|
} while (*dp++);
|
} while (*dp++);
|
/* Continue the loop in case NAME appears again. */
|
/* Continue the loop in case NAME appears again. */
|
} else {
|
} else {
|
++ep;
|
++ep;
|
}
|
}
|
}
|
}
|
UNLOCK;
|
UNLOCK;
|
return 0;
|
return 0;
|
}
|
}
|
|
|
/* The `clearenv' was planned to be added to POSIX.1 but probably
|
/* The `clearenv' was planned to be added to POSIX.1 but probably
|
never made it. Nevertheless the POSIX.9 standard (POSIX bindings
|
never made it. Nevertheless the POSIX.9 standard (POSIX bindings
|
for Fortran 77) requires this function. */
|
for Fortran 77) requires this function. */
|
int clearenv (void)
|
int clearenv (void)
|
{
|
{
|
LOCK;
|
LOCK;
|
if (__environ == last_environ && __environ != NULL) {
|
if (__environ == last_environ && __environ != NULL) {
|
/* We allocated this environment so we can free it. */
|
/* We allocated this environment so we can free it. */
|
free (__environ);
|
free (__environ);
|
last_environ = NULL;
|
last_environ = NULL;
|
}
|
}
|
/* Clear the environment pointer removes the whole environment. */
|
/* Clear the environment pointer removes the whole environment. */
|
__environ = NULL;
|
__environ = NULL;
|
UNLOCK;
|
UNLOCK;
|
return 0;
|
return 0;
|
}
|
}
|
|
|
/* Put STRING, which is of the form "NAME=VALUE", in the environment. */
|
/* Put STRING, which is of the form "NAME=VALUE", in the environment. */
|
int putenv (char *string)
|
int putenv (char *string)
|
{
|
{
|
int result;
|
int result;
|
const char *const name_end = strchr (string, '=');
|
const char *const name_end = strchr (string, '=');
|
|
|
if (name_end != NULL) {
|
if (name_end != NULL) {
|
char *name = strndup(string, name_end - string);
|
char *name = strndup(string, name_end - string);
|
result = __add_to_environ (name, NULL, string, 1);
|
result = __add_to_environ (name, NULL, string, 1);
|
free(name);
|
free(name);
|
return(result);
|
return(result);
|
}
|
}
|
unsetenv (string);
|
unsetenv (string);
|
return 0;
|
return 0;
|
}
|
}
|
|
|
|
|