/ stdlib / FreeBSD / getenv.c
getenv.c
  1  /*
  2   * Copyright (c) 1987, 1993
  3   *	The Regents of the University of California.  All rights reserved.
  4   *
  5   * Redistribution and use in source and binary forms, with or without
  6   * modification, are permitted provided that the following conditions
  7   * are met:
  8   * 1. Redistributions of source code must retain the above copyright
  9   *    notice, this list of conditions and the following disclaimer.
 10   * 2. Redistributions in binary form must reproduce the above copyright
 11   *    notice, this list of conditions and the following disclaimer in the
 12   *    documentation and/or other materials provided with the distribution.
 13   * 4. Neither the name of the University nor the names of its contributors
 14   *    may be used to endorse or promote products derived from this software
 15   *    without specific prior written permission.
 16   *
 17   * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 18   * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 19   * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 20   * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 21   * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 22   * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 23   * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 24   * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 25   * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 26   * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 27   * SUCH DAMAGE.
 28   */
 29  
 30  #if defined(LIBC_SCCS) && !defined(lint)
 31  static char sccsid[] = "@(#)getenv.c	8.1 (Berkeley) 6/4/93";
 32  #endif /* LIBC_SCCS and not lint */
 33  #include <sys/cdefs.h>
 34  __FBSDID("$FreeBSD: src/lib/libc/stdlib/getenv.c,v 1.8 2007/05/01 16:02:41 ache Exp $");
 35  
 36  #include <os/lock_private.h>
 37  #include <stdlib.h>
 38  #include <stddef.h>
 39  #include <string.h>
 40  #include <crt_externs.h>
 41  
 42  __private_extern__ char *__findenv_locked(const char *, int *, char **);
 43  __private_extern__ void __environ_lock(void);
 44  __private_extern__ void __environ_unlock(void);
 45  
 46  /*
 47   * __findenv_locked --
 48   *	Returns pointer to value associated with name, if any, else NULL.
 49   *	Sets offset to be the offset of the name/value combination in the
 50   *	environmental array, for use by setenv(3) and unsetenv(3).
 51   *	Explicitly removes '=' in argument name.
 52   *
 53   *	This routine *should* be a static; don't use it.
 54   */
 55  __private_extern__ char *
 56  __findenv_locked(name, offset, environ)
 57  	const char *name;
 58  	int *offset;
 59  	char **environ;
 60  {
 61  	int len, i;
 62  	const char *np;
 63  	char **p, *cp;
 64  
 65  	if (name == NULL || environ == NULL)
 66  		return (NULL);
 67  	for (np = name; *np && *np != '='; ++np)
 68  		continue;
 69  	len = np - name;
 70  	for (p = environ; (cp = *p) != NULL; ++p) {
 71  		for (np = name, i = len; i && *cp; i--)
 72  			if (*cp++ != *np++)
 73  				break;
 74  		if (i == 0 && *cp++ == '=') {
 75  			*offset = p - environ;
 76  			return (cp);
 77  		}
 78  	}
 79  	return (NULL);
 80  }
 81  
 82  static os_unfair_lock __environ_lock_obj = OS_UNFAIR_LOCK_INIT;
 83  __private_extern__ void
 84  __environ_lock(void)
 85  {
 86  	os_unfair_lock_lock_with_options(
 87  			&__environ_lock_obj, OS_UNFAIR_LOCK_DATA_SYNCHRONIZATION);
 88  }
 89  __private_extern__ void
 90  __environ_unlock(void)
 91  {
 92  	os_unfair_lock_unlock(&__environ_lock_obj);
 93  }
 94  __private_extern__ void
 95  __environ_lock_fork_child(void)
 96  {
 97  	__environ_lock_obj = OS_UNFAIR_LOCK_INIT;
 98  }
 99  
100  /*
101   * _getenvp -- SPI using an arbitrary pointer to string array (the array must
102   * have been created with malloc) and an env state, created by _allocenvstate().
103   *	Returns ptr to value associated with name, if any, else NULL.
104   */
105  char *
106  _getenvp(const char *name, char ***envp, void *state __unused)
107  {
108  	// envp is passed as an argument, so the lock is not protecting everything
109  	int offset;
110  	__environ_lock();
111  	char *result = (__findenv_locked(name, &offset, *envp));
112  	__environ_unlock();
113  	return result;
114  }
115  
116  /*
117   * getenv --
118   *	Returns ptr to value associated with name, if any, else NULL.
119   */
120  char *
121  getenv(name)
122  	const char *name;
123  {
124  	int offset;
125  	__environ_lock();
126  	char *result = __findenv_locked(name, &offset, *_NSGetEnviron());
127  	__environ_unlock();
128  	return result;
129  }