Home | History | Annotate | Line # | Download | only in stdlib
getenv.c revision 1.20
      1  1.20  christos /*	$NetBSD: getenv.c,v 1.20 2010/09/23 17:30:49 christos 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.20  christos __RCSID("$NetBSD: getenv.c,v 1.20 2010/09/23 17:30:49 christos 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 <bitstring.h>
     47  1.20  christos #include "reentrant.h"
     48   1.8  christos #include "local.h"
     49  1.11    kleink 
     50  1.16   thorpej #ifdef _REENTRANT
     51  1.11    kleink rwlock_t __environ_lock = RWLOCK_INITIALIZER;
     52  1.11    kleink #endif
     53  1.20  christos bitstr_t *__environ_malloced;
     54  1.20  christos static size_t environ_bitlen;
     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.20  christos 	bitstr_t *s;
    108  1.20  christos 	size_t nl;
    109  1.20  christos 
    110  1.20  christos 	if (offset == -1) {
    111  1.20  christos 		char **ptr;
    112  1.20  christos 		for (ptr = environ, offset = 0; *ptr; ptr++)
    113  1.20  christos 			offset++;
    114  1.20  christos 	}
    115  1.20  christos 	nl = bitstr_size(offset + 2);
    116  1.20  christos 	if (__environ_malloced == NULL) {
    117  1.20  christos 		s = malloc(nl);
    118  1.20  christos 	} else if (environ_bitlen < nl)
    119  1.20  christos 		s = realloc(__environ_malloced, nl);
    120  1.20  christos 	else
    121  1.20  christos 		return 0;
    122  1.20  christos 
    123  1.20  christos 	if (s == NULL)
    124  1.20  christos 		return -1;
    125  1.20  christos 
    126  1.20  christos 	(void)memset(&s[environ_bitlen], 0, nl - environ_bitlen);
    127  1.20  christos 	environ_bitlen = nl;
    128  1.20  christos 	__environ_malloced = s;
    129  1.20  christos 
    130  1.20  christos 	return 0;
    131  1.20  christos }
    132  1.20  christos 
    133   1.1       cgd /*
    134   1.4       jtc  * __findenv --
    135   1.1       cgd  *	Returns pointer to value associated with name, if any, else NULL.
    136   1.1       cgd  *	Sets offset to be the offset of the name/value combination in the
    137   1.1       cgd  *	environmental array, for use by setenv(3) and unsetenv(3).
    138   1.1       cgd  *	Explicitly removes '=' in argument name.
    139   1.1       cgd  *
    140   1.1       cgd  *	This routine *should* be a static; don't use it.
    141   1.1       cgd  */
    142   1.1       cgd char *
    143  1.18  christos __findenv(const char *name, int *offset)
    144   1.1       cgd {
    145  1.12  christos 	size_t len;
    146  1.10     perry 	const char *np;
    147  1.10     perry 	char **p, *c;
    148   1.1       cgd 
    149   1.9     perry 	if (name == NULL || environ == NULL)
    150  1.18  christos 		return NULL;
    151   1.9     perry 	for (np = name; *np && *np != '='; ++np)
    152   1.9     perry 		continue;
    153   1.9     perry 	len = np - name;
    154   1.9     perry 	for (p = environ; (c = *p) != NULL; ++p)
    155   1.9     perry 		if (strncmp(c, name, len) == 0 && c[len] == '=') {
    156   1.9     perry 			*offset = p - environ;
    157  1.18  christos 			return c + len + 1;
    158   1.9     perry 		}
    159  1.19  christos 	*offset = p - environ;
    160  1.18  christos 	return NULL;
    161   1.1       cgd }
    162