/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include "primpl.h" #include "prmem.h" #if defined(XP_UNIX) # include # if defined(DARWIN) # if defined(HAVE_CRT_EXTERNS_H) # include # endif /* HAVE_CRT_EXTERNS_H */ # else /* DARWIN */ PR_IMPORT_DATA(char**) environ; # endif /* DARWIN */ #endif /* XP_UNIX */ #if !defined(HAVE_SECURE_GETENV) && defined(HAVE___SECURE_GETENV) # define secure_getenv __secure_getenv # define HAVE_SECURE_GETENV 1 #endif /* Lock used to lock the environment */ #if defined(_PR_NO_PREEMPT) # define _PR_NEW_LOCK_ENV() # define _PR_DELETE_LOCK_ENV() # define _PR_LOCK_ENV() # define _PR_UNLOCK_ENV() #elif defined(_PR_LOCAL_THREADS_ONLY) extern _PRCPU* _pr_primordialCPU; static PRIntn _is; # define _PR_NEW_LOCK_ENV() # define _PR_DELETE_LOCK_ENV() # define _PR_LOCK_ENV() \ if (_pr_primordialCPU) _PR_INTSOFF(_is); # define _PR_UNLOCK_ENV() \ if (_pr_primordialCPU) _PR_INTSON(_is); #else static PRLock* _pr_envLock = NULL; # define _PR_NEW_LOCK_ENV() \ { _pr_envLock = PR_NewLock(); } # define _PR_DELETE_LOCK_ENV() \ { \ if (_pr_envLock) { \ PR_DestroyLock(_pr_envLock); \ _pr_envLock = NULL; \ } \ } # define _PR_LOCK_ENV() \ { \ if (_pr_envLock) PR_Lock(_pr_envLock); \ } # define _PR_UNLOCK_ENV() \ { \ if (_pr_envLock) PR_Unlock(_pr_envLock); \ } #endif /************************************************************************/ void _PR_InitEnv(void) { _PR_NEW_LOCK_ENV(); } void _PR_CleanupEnv(void) { _PR_DELETE_LOCK_ENV(); } PR_IMPLEMENT(char*) PR_GetEnv(const char* var) { char* ev; if (!_pr_initialized) { _PR_ImplicitInitialization(); } _PR_LOCK_ENV(); ev = _PR_MD_GET_ENV(var); _PR_UNLOCK_ENV(); return ev; } PR_IMPLEMENT(char*) PR_GetEnvSecure(const char* var) { #ifdef HAVE_SECURE_GETENV char* ev; if (!_pr_initialized) { _PR_ImplicitInitialization(); } _PR_LOCK_ENV(); ev = secure_getenv(var); _PR_UNLOCK_ENV(); return ev; #else # ifdef XP_UNIX /* ** Fall back to checking uids and gids. This won't detect any other ** privilege-granting mechanisms the platform may have. This also ** can't detect the case where the process already called ** setuid(geteuid()) and/or setgid(getegid()). */ if (getuid() != geteuid() || getgid() != getegid()) { return NULL; } # endif /* XP_UNIX */ return PR_GetEnv(var); #endif /* HAVE_SECURE_GETENV */ } PR_IMPLEMENT(PRStatus) PR_SetEnv(const char* string) { PRIntn result; if (!_pr_initialized) { _PR_ImplicitInitialization(); } if (!strchr(string, '=')) { return (PR_FAILURE); } _PR_LOCK_ENV(); result = _PR_MD_PUT_ENV((char*)string); _PR_UNLOCK_ENV(); return result ? PR_FAILURE : PR_SUCCESS; } #if defined(XP_UNIX) && (!defined(DARWIN) || defined(HAVE_CRT_EXTERNS_H)) PR_IMPLEMENT(char**) PR_DuplicateEnvironment(void) { char **the_environ, **result, **end, **src, **dst; _PR_LOCK_ENV(); # ifdef DARWIN the_environ = *(_NSGetEnviron()); # else the_environ = environ; # endif for (end = the_environ; *end != NULL; end++) /* empty loop body */; result = (char**)PR_Malloc(sizeof(char*) * (end - the_environ + 1)); if (result != NULL) { for (src = the_environ, dst = result; src != end; src++, dst++) { size_t len; len = strlen(*src) + 1; *dst = PR_Malloc(len); if (*dst == NULL) { /* Allocation failed. Must clean up the half-copied env. */ char** to_delete; for (to_delete = result; to_delete != dst; to_delete++) { PR_Free(*to_delete); } PR_Free(result); result = NULL; goto out; } memcpy(*dst, *src, len); } *dst = NULL; } out: _PR_UNLOCK_ENV(); return result; } #else /* This platform doesn't support raw access to the environ block. */ PR_IMPLEMENT(char**) PR_DuplicateEnvironment(void) { return NULL; } #endif