Home | History | Annotate | Line # | Download | only in stdlib
getenv.c revision 1.25
      1  1.25      tron /*	$NetBSD: getenv.c,v 1.25 2010/10/02 10:51:07 tron 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.25      tron __RCSID("$NetBSD: getenv.c,v 1.25 2010/10/02 10:51:07 tron 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.20  christos 	size_t nl;
    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.22  christos 	nl = offset + 2; 	/* one for potentially new entry one for NULL */
    117  1.24  christos 	if (nl <= environ_malloced_len && saveenv == environ)
    118  1.20  christos 		return 0;
    119  1.20  christos 
    120  1.24  christos 	if (saveenv == environ) {		/* just increase size */
    121  1.24  christos 		if ((p = realloc(saveenv, nl * sizeof(*p))) == NULL)
    122  1.24  christos 			return -1;
    123  1.25      tron 		(void)memset(&p[environ_malloced_len], 0,
    124  1.25      tron 		    (nl - environ_malloced_len) * sizeof(*p));
    125  1.24  christos 		saveenv = p;
    126  1.24  christos 	} else {				/* get new space */
    127  1.24  christos 		free(saveenv);
    128  1.24  christos 		if ((saveenv = malloc(nl * sizeof(*saveenv))) == NULL)
    129  1.24  christos 			return -1;
    130  1.24  christos 		(void)memcpy(saveenv, environ, (nl - 2) * sizeof(*saveenv));
    131  1.25      tron 		saveenv[nl - 2] = NULL;
    132  1.25      tron 		saveenv[nl - 1] = NULL;
    133  1.24  christos 	}
    134  1.24  christos 	environ = saveenv;
    135  1.24  christos 
    136  1.22  christos 	p = realloc(__environ_malloced, nl * sizeof(*p));
    137  1.21      tron 	if (p == NULL)
    138  1.20  christos 		return -1;
    139  1.20  christos 
    140  1.21      tron 	(void)memset(&p[environ_malloced_len], 0,
    141  1.22  christos 	    (nl - environ_malloced_len) * sizeof(*p));
    142  1.21      tron 	environ_malloced_len = nl;
    143  1.21      tron 	__environ_malloced = p;
    144  1.20  christos 
    145  1.20  christos 	return 0;
    146  1.20  christos }
    147  1.20  christos 
    148   1.1       cgd /*
    149   1.4       jtc  * __findenv --
    150   1.1       cgd  *	Returns pointer to value associated with name, if any, else NULL.
    151   1.1       cgd  *	Sets offset to be the offset of the name/value combination in the
    152   1.1       cgd  *	environmental array, for use by setenv(3) and unsetenv(3).
    153   1.1       cgd  *	Explicitly removes '=' in argument name.
    154   1.1       cgd  *
    155   1.1       cgd  *	This routine *should* be a static; don't use it.
    156   1.1       cgd  */
    157   1.1       cgd char *
    158  1.18  christos __findenv(const char *name, int *offset)
    159   1.1       cgd {
    160  1.12  christos 	size_t len;
    161  1.10     perry 	const char *np;
    162  1.10     perry 	char **p, *c;
    163   1.1       cgd 
    164   1.9     perry 	if (name == NULL || environ == NULL)
    165  1.18  christos 		return NULL;
    166   1.9     perry 	for (np = name; *np && *np != '='; ++np)
    167   1.9     perry 		continue;
    168   1.9     perry 	len = np - name;
    169   1.9     perry 	for (p = environ; (c = *p) != NULL; ++p)
    170   1.9     perry 		if (strncmp(c, name, len) == 0 && c[len] == '=') {
    171  1.24  christos 			*offset = (int)(p - environ);
    172  1.18  christos 			return c + len + 1;
    173   1.9     perry 		}
    174  1.24  christos 	*offset = (int)(p - environ);
    175  1.18  christos 	return NULL;
    176   1.1       cgd }
    177