X-Git-Url: https://git.saurik.com/apple/libc.git/blobdiff_plain/ad3c9f2af814c84582fdd1649e49ec4f68572c5a..aa54d2fad3d9038b43475aa93c76795c5141a993:/stdlib/FreeBSD/getenv.c diff --git a/stdlib/FreeBSD/getenv.c b/stdlib/FreeBSD/getenv.c index 47b60df..c012578 100644 --- a/stdlib/FreeBSD/getenv.c +++ b/stdlib/FreeBSD/getenv.c @@ -33,15 +33,18 @@ static char sccsid[] = "@(#)getenv.c 8.1 (Berkeley) 6/4/93"; #include __FBSDID("$FreeBSD: src/lib/libc/stdlib/getenv.c,v 1.8 2007/05/01 16:02:41 ache Exp $"); +#include #include #include #include #include -__private_extern__ char *__findenv(const char *, int *, char **); +__private_extern__ char *__findenv_locked(const char *, int *, char **); +__private_extern__ void __environ_lock(void); +__private_extern__ void __environ_unlock(void); /* - * __findenv -- + * __findenv_locked -- * Returns pointer to value associated with name, if any, else NULL. * Sets offset to be the offset of the name/value combination in the * environmental array, for use by setenv(3) and unsetenv(3). @@ -50,7 +53,7 @@ __private_extern__ char *__findenv(const char *, int *, char **); * This routine *should* be a static; don't use it. */ __private_extern__ char * -__findenv(name, offset, environ) +__findenv_locked(name, offset, environ) const char *name; int *offset; char **environ; @@ -76,6 +79,24 @@ __findenv(name, offset, environ) return (NULL); } +static os_unfair_lock __environ_lock_obj = OS_UNFAIR_LOCK_INIT; +__private_extern__ void +__environ_lock(void) +{ + os_unfair_lock_lock_with_options( + &__environ_lock_obj, OS_UNFAIR_LOCK_DATA_SYNCHRONIZATION); +} +__private_extern__ void +__environ_unlock(void) +{ + os_unfair_lock_unlock(&__environ_lock_obj); +} +__private_extern__ void +__environ_lock_fork_child(void) +{ + __environ_lock_obj = OS_UNFAIR_LOCK_INIT; +} + /* * _getenvp -- SPI using an arbitrary pointer to string array (the array must * have been created with malloc) and an env state, created by _allocenvstate(). @@ -84,9 +105,12 @@ __findenv(name, offset, environ) char * _getenvp(const char *name, char ***envp, void *state __unused) { + // envp is passed as an argument, so the lock is not protecting everything int offset; - - return (__findenv(name, &offset, *envp)); + __environ_lock(); + char *result = (__findenv_locked(name, &offset, *envp)); + __environ_unlock(); + return result; } /* @@ -98,6 +122,8 @@ getenv(name) const char *name; { int offset; - - return (__findenv(name, &offset, *_NSGetEnviron())); + __environ_lock(); + char *result = __findenv_locked(name, &offset, *_NSGetEnviron()); + __environ_unlock(); + return result; }