Home | History | Annotate | Line # | Download | only in stdlib
getenv.c revision 1.31
      1  1.31     enami /*	$NetBSD: getenv.c,v 1.31 2010/11/10 02:33:49 enami Exp $	*/
      2   1.6   thorpej 
      3   1.1       cgd /*
      4   1.9     perry  * Copyright (c) 1987, 1993
      5   1.9     perry  *	The Regents of the University of California.  All rights reserved.
      6   1.1       cgd  *
      7   1.1       cgd  * Redistribution and use in source and binary forms, with or without
      8   1.1       cgd  * modification, are permitted provided that the following conditions
      9   1.1       cgd  * are met:
     10   1.1       cgd  * 1. Redistributions of source code must retain the above copyright
     11   1.1       cgd  *    notice, this list of conditions and the following disclaimer.
     12   1.1       cgd  * 2. Redistributions in binary form must reproduce the above copyright
     13   1.1       cgd  *    notice, this list of conditions and the following disclaimer in the
     14   1.1       cgd  *    documentation and/or other materials provided with the distribution.
     15  1.17       agc  * 3. Neither the name of the University nor the names of its contributors
     16   1.1       cgd  *    may be used to endorse or promote products derived from this software
     17   1.1       cgd  *    without specific prior written permission.
     18   1.1       cgd  *
     19   1.1       cgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
     20   1.1       cgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     21   1.1       cgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     22   1.1       cgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
     23   1.1       cgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     24   1.1       cgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     25   1.1       cgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     26   1.1       cgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     27   1.1       cgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     28   1.1       cgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     29   1.1       cgd  * SUCH DAMAGE.
     30   1.1       cgd  */
     31   1.1       cgd 
     32   1.8  christos #include <sys/cdefs.h>
     33   1.1       cgd #if defined(LIBC_SCCS) && !defined(lint)
     34   1.6   thorpej #if 0
     35   1.9     perry static char sccsid[] = "@(#)getenv.c	8.1 (Berkeley) 6/4/93";
     36   1.6   thorpej #else
     37  1.31     enami __RCSID("$NetBSD: getenv.c,v 1.31 2010/11/10 02:33:49 enami Exp $");
     38   1.6   thorpej #endif
     39   1.1       cgd #endif /* LIBC_SCCS and not lint */
     40   1.1       cgd 
     41  1.18  christos #include "namespace.h"
     42  1.13     lukem #include <assert.h>
     43  1.13     lukem #include <errno.h>
     44   1.1       cgd #include <stdlib.h>
     45   1.1       cgd #include <string.h>
     46  1.20  christos #include "reentrant.h"
     47   1.8  christos #include "local.h"
     48  1.11    kleink 
     49  1.16   thorpej #ifdef _REENTRANT
     50  1.11    kleink rwlock_t __environ_lock = RWLOCK_INITIALIZER;
     51  1.11    kleink #endif
     52  1.21      tron char **__environ_malloced;
     53  1.24  christos static char **saveenv;
     54  1.21      tron static size_t environ_malloced_len;
     55   1.1       cgd 
     56  1.18  christos __weak_alias(getenv_r, _getenv_r)
     57  1.18  christos 
     58   1.1       cgd /*
     59   1.1       cgd  * getenv --
     60   1.1       cgd  *	Returns ptr to value associated with name, if any, else NULL.
     61  1.18  christos  *	XXX: we cannot use getenv_r to implement this, because getenv()
     62  1.18  christos  *	cannot use a shared buffer, because if it did, subsequent calls
     63  1.18  christos  *	to getenv would trash previous results.
     64   1.1       cgd  */
     65   1.1       cgd char *
     66  1.18  christos getenv(const char *name)
     67  1.18  christos {
     68  1.18  christos 	int offset;
     69  1.18  christos 	char *result;
     70  1.18  christos 
     71  1.18  christos 	_DIAGASSERT(name != NULL);
     72  1.18  christos 
     73  1.18  christos 	rwlock_rdlock(&__environ_lock);
     74  1.18  christos 	result = __findenv(name, &offset);
     75  1.18  christos 	rwlock_unlock(&__environ_lock);
     76  1.18  christos 	return result;
     77  1.18  christos }
     78  1.18  christos 
     79  1.18  christos int
     80  1.18  christos getenv_r(const char *name, char *buf, size_t len)
     81   1.1       cgd {
     82   1.1       cgd 	int offset;
     83  1.11    kleink 	char *result;
     84  1.18  christos 	int rv = -1;
     85  1.13     lukem 
     86  1.13     lukem 	_DIAGASSERT(name != NULL);
     87   1.1       cgd 
     88  1.11    kleink 	rwlock_rdlock(&__environ_lock);
     89  1.11    kleink 	result = __findenv(name, &offset);
     90  1.18  christos 	if (result == NULL) {
     91  1.18  christos 		errno = ENOENT;
     92  1.18  christos 		goto out;
     93  1.18  christos 	}
     94  1.18  christos 	if (strlcpy(buf, result, len) >= len) {
     95  1.18  christos 		errno = ERANGE;
     96  1.18  christos 		goto out;
     97  1.18  christos 	}
     98  1.18  christos 	rv = 0;
     99  1.18  christos out:
    100  1.11    kleink 	rwlock_unlock(&__environ_lock);
    101  1.18  christos 	return rv;
    102   1.1       cgd }
    103   1.1       cgd 
    104  1.20  christos int
    105  1.20  christos __allocenv(int offset)
    106  1.20  christos {
    107  1.21      tron 	char **p;
    108  1.26      tron 	size_t required_len, new_len;
    109  1.20  christos 
    110  1.24  christos 	if (offset == -1 || saveenv != environ) {
    111  1.20  christos 		char **ptr;
    112  1.21      tron 		for (ptr = environ, offset = 0; *ptr != NULL; ptr++)
    113  1.20  christos 			offset++;
    114  1.20  christos 	}
    115  1.21      tron 
    116  1.26      tron 	/* one for potentially new entry one for NULL */
    117  1.26      tron 	required_len = offset + 2;
    118  1.26      tron 
    119  1.26      tron 	if (required_len <= environ_malloced_len && saveenv == environ)
    120  1.20  christos 		return 0;
    121  1.20  christos 
    122  1.28     enami 	/* Double the size of the arrays until we meet the requirement. */
    123  1.28     enami 	new_len = environ_malloced_len ? environ_malloced_len : 16;
    124  1.26      tron 	while (new_len < required_len)
    125  1.28     enami 		new_len <<= 1;
    126  1.26      tron 
    127  1.24  christos 	if (saveenv == environ) {		/* just increase size */
    128  1.26      tron 		if ((p = realloc(saveenv, new_len * sizeof(*p))) == NULL)
    129  1.24  christos 			return -1;
    130  1.25      tron 		(void)memset(&p[environ_malloced_len], 0,
    131  1.26      tron 		    (new_len - environ_malloced_len) * sizeof(*p));
    132  1.24  christos 		saveenv = p;
    133  1.24  christos 	} else {				/* get new space */
    134  1.24  christos 		free(saveenv);
    135  1.26      tron 		if ((saveenv = malloc(new_len * sizeof(*saveenv))) == NULL)
    136  1.24  christos 			return -1;
    137  1.26      tron 		(void)memcpy(saveenv, environ,
    138  1.26      tron 		    (required_len - 2) * sizeof(*saveenv));
    139  1.26      tron 		(void)memset(&saveenv[required_len - 2], 0,
    140  1.26      tron 		    (new_len - (required_len - 2)) * sizeof(*saveenv));
    141  1.24  christos 	}
    142  1.24  christos 	environ = saveenv;
    143  1.24  christos 
    144  1.26      tron 	p = realloc(__environ_malloced, new_len * sizeof(*p));
    145  1.21      tron 	if (p == NULL)
    146  1.20  christos 		return -1;
    147  1.20  christos 
    148  1.21      tron 	(void)memset(&p[environ_malloced_len], 0,
    149  1.26      tron 	    (new_len - environ_malloced_len) * sizeof(*p));
    150  1.26      tron 	environ_malloced_len = new_len;
    151  1.21      tron 	__environ_malloced = p;
    152  1.20  christos 
    153  1.20  christos 	return 0;
    154  1.20  christos }
    155  1.20  christos 
    156   1.1       cgd /*
    157  1.29  christos  * Handle the case where a program tried to cleanup the environment
    158  1.29  christos  * by setting *environ = NULL; we attempt to cleanup all the malloced
    159  1.29  christos  * environ entries and we make sure that the entry following the new
    160  1.29  christos  * entry is NULL.
    161  1.29  christos  */
    162  1.29  christos void
    163  1.29  christos __scrubenv(int offset)
    164  1.29  christos {
    165  1.29  christos 	if (environ[++offset] == NULL)
    166  1.29  christos 		return;
    167  1.29  christos 
    168  1.30  christos 	for (; environ[offset]; offset++)
    169  1.31     enami 		if (environ[offset] == __environ_malloced[offset]) {
    170  1.31     enami 			free(__environ_malloced[offset]);
    171  1.31     enami 			environ[offset] = __environ_malloced[offset] = NULL;
    172  1.31     enami 		}
    173  1.29  christos 
    174  1.29  christos 	environ[offset] = __environ_malloced[offset] = NULL;
    175  1.29  christos }
    176  1.29  christos 
    177  1.29  christos /*
    178   1.4       jtc  * __findenv --
    179   1.1       cgd  *	Returns pointer to value associated with name, if any, else NULL.
    180   1.1       cgd  *	Sets offset to be the offset of the name/value combination in the
    181   1.1       cgd  *	environmental array, for use by setenv(3) and unsetenv(3).
    182   1.1       cgd  *	Explicitly removes '=' in argument name.
    183   1.1       cgd  *
    184   1.1       cgd  *	This routine *should* be a static; don't use it.
    185   1.1       cgd  */
    186   1.1       cgd char *
    187  1.18  christos __findenv(const char *name, int *offset)
    188   1.1       cgd {
    189  1.12  christos 	size_t len;
    190  1.10     perry 	const char *np;
    191  1.10     perry 	char **p, *c;
    192   1.1       cgd 
    193   1.9     perry 	if (name == NULL || environ == NULL)
    194  1.18  christos 		return NULL;
    195   1.9     perry 	for (np = name; *np && *np != '='; ++np)
    196   1.9     perry 		continue;
    197   1.9     perry 	len = np - name;
    198   1.9     perry 	for (p = environ; (c = *p) != NULL; ++p)
    199   1.9     perry 		if (strncmp(c, name, len) == 0 && c[len] == '=') {
    200  1.24  christos 			*offset = (int)(p - environ);
    201  1.18  christos 			return c + len + 1;
    202   1.9     perry 		}
    203  1.24  christos 	*offset = (int)(p - environ);
    204  1.18  christos 	return NULL;
    205   1.1       cgd }
    206