| /* |
| * msvcrt.dll environment functions |
| * |
| * Copyright 1996,1998 Marcus Meissner |
| * Copyright 1996 Jukka Iivonen |
| * Copyright 1997,2000 Uwe Bonnes |
| * Copyright 2000 Jon Griffiths |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| #include "wine/unicode.h" |
| #include "msvcrt.h" |
| #include "wine/debug.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(msvcrt); |
| |
| /********************************************************************* |
| * getenv (MSVCRT.@) |
| */ |
| char * CDECL MSVCRT_getenv(const char *name) |
| { |
| char **environ; |
| unsigned int length=strlen(name); |
| |
| for (environ = MSVCRT__environ; *environ; environ++) |
| { |
| char *str = *environ; |
| char *pos = strchr(str,'='); |
| if (pos && ((pos - str) == length) && !strncasecmp(str,name,length)) |
| { |
| TRACE("(%s): got %s\n", debugstr_a(name), debugstr_a(pos + 1)); |
| return pos + 1; |
| } |
| } |
| return NULL; |
| } |
| |
| /********************************************************************* |
| * _wgetenv (MSVCRT.@) |
| */ |
| MSVCRT_wchar_t * CDECL MSVCRT__wgetenv(const MSVCRT_wchar_t *name) |
| { |
| MSVCRT_wchar_t **environ; |
| unsigned int length=strlenW(name); |
| |
| /* Initialize the _wenviron array if it's not already created. */ |
| if (!MSVCRT__wenviron) |
| MSVCRT__wenviron = msvcrt_SnapshotOfEnvironmentW(NULL); |
| |
| for (environ = MSVCRT__wenviron; *environ; environ++) |
| { |
| MSVCRT_wchar_t *str = *environ; |
| MSVCRT_wchar_t *pos = strchrW(str,'='); |
| if (pos && ((pos - str) == length) && !strncmpiW(str,name,length)) |
| { |
| TRACE("(%s): got %s\n", debugstr_w(name), debugstr_w(pos + 1)); |
| return pos + 1; |
| } |
| } |
| return NULL; |
| } |
| |
| /********************************************************************* |
| * _putenv (MSVCRT.@) |
| */ |
| int CDECL _putenv(const char *str) |
| { |
| char *name, *value; |
| char *dst; |
| int ret; |
| |
| TRACE("%s\n", debugstr_a(str)); |
| |
| if (!str) |
| return -1; |
| |
| name = HeapAlloc(GetProcessHeap(), 0, strlen(str) + 1); |
| if (!name) |
| return -1; |
| dst = name; |
| while (*str && *str != '=') |
| *dst++ = *str++; |
| if (!*str++) |
| { |
| ret = -1; |
| goto finish; |
| } |
| *dst++ = '\0'; |
| value = dst; |
| while (*str) |
| *dst++ = *str++; |
| *dst = '\0'; |
| |
| ret = SetEnvironmentVariableA(name, value[0] ? value : NULL) ? 0 : -1; |
| |
| /* _putenv returns success on deletion of nonexistent variable, unlike [Rtl]SetEnvironmentVariable */ |
| if ((ret == -1) && (GetLastError() == ERROR_ENVVAR_NOT_FOUND)) ret = 0; |
| |
| MSVCRT__environ = msvcrt_SnapshotOfEnvironmentA(MSVCRT__environ); |
| /* Update the __p__wenviron array only when already initialized */ |
| if (MSVCRT__wenviron) |
| MSVCRT__wenviron = msvcrt_SnapshotOfEnvironmentW(MSVCRT__wenviron); |
| |
| finish: |
| HeapFree(GetProcessHeap(), 0, name); |
| return ret; |
| } |
| |
| /********************************************************************* |
| * _wputenv (MSVCRT.@) |
| */ |
| int CDECL _wputenv(const MSVCRT_wchar_t *str) |
| { |
| MSVCRT_wchar_t *name, *value; |
| MSVCRT_wchar_t *dst; |
| int ret; |
| |
| TRACE("%s\n", debugstr_w(str)); |
| |
| if (!str) |
| return -1; |
| name = HeapAlloc(GetProcessHeap(), 0, (strlenW(str) + 1) * sizeof(MSVCRT_wchar_t)); |
| if (!name) |
| return -1; |
| dst = name; |
| while (*str && *str != '=') |
| *dst++ = *str++; |
| if (!*str++) |
| { |
| ret = -1; |
| goto finish; |
| } |
| *dst++ = 0; |
| value = dst; |
| while (*str) |
| *dst++ = *str++; |
| *dst = 0; |
| |
| ret = SetEnvironmentVariableW(name, value[0] ? value : NULL) ? 0 : -1; |
| |
| /* _putenv returns success on deletion of nonexistent variable, unlike [Rtl]SetEnvironmentVariable */ |
| if ((ret == -1) && (GetLastError() == ERROR_ENVVAR_NOT_FOUND)) ret = 0; |
| |
| MSVCRT__environ = msvcrt_SnapshotOfEnvironmentA(MSVCRT__environ); |
| MSVCRT__wenviron = msvcrt_SnapshotOfEnvironmentW(MSVCRT__wenviron); |
| |
| finish: |
| HeapFree(GetProcessHeap(), 0, name); |
| return ret; |
| } |
| |
| /********************************************************************* |
| * _putenv_s (MSVCRT.@) |
| */ |
| int CDECL _putenv_s(const char *name, const char *value) |
| { |
| int ret; |
| |
| TRACE("%s %s\n", debugstr_a(name), debugstr_a(value)); |
| |
| if (!MSVCRT_CHECK_PMT(name != NULL)) return -1; |
| if (!MSVCRT_CHECK_PMT(value != NULL)) return -1; |
| |
| ret = SetEnvironmentVariableA(name, value[0] ? value : NULL) ? 0 : -1; |
| |
| /* _putenv returns success on deletion of nonexistent variable, unlike [Rtl]SetEnvironmentVariable */ |
| if ((ret == -1) && (GetLastError() == ERROR_ENVVAR_NOT_FOUND)) ret = 0; |
| |
| MSVCRT__environ = msvcrt_SnapshotOfEnvironmentA(MSVCRT__environ); |
| MSVCRT__wenviron = msvcrt_SnapshotOfEnvironmentW(MSVCRT__wenviron); |
| |
| return ret; |
| } |
| |
| /********************************************************************* |
| * _wputenv_s (MSVCRT.@) |
| */ |
| int CDECL _wputenv_s(const MSVCRT_wchar_t *name, const MSVCRT_wchar_t *value) |
| { |
| int ret; |
| |
| TRACE("%s %s\n", debugstr_w(name), debugstr_w(value)); |
| |
| if (!MSVCRT_CHECK_PMT(name != NULL)) return -1; |
| if (!MSVCRT_CHECK_PMT(value != NULL)) return -1; |
| |
| ret = SetEnvironmentVariableW(name, value[0] ? value : NULL) ? 0 : -1; |
| |
| /* _putenv returns success on deletion of nonexistent variable, unlike [Rtl]SetEnvironmentVariable */ |
| if ((ret == -1) && (GetLastError() == ERROR_ENVVAR_NOT_FOUND)) ret = 0; |
| |
| MSVCRT__environ = msvcrt_SnapshotOfEnvironmentA(MSVCRT__environ); |
| MSVCRT__wenviron = msvcrt_SnapshotOfEnvironmentW(MSVCRT__wenviron); |
| |
| return ret; |
| } |
| |
| /****************************************************************** |
| * _dupenv_s (MSVCRT.@) |
| */ |
| int CDECL _dupenv_s(char **buffer, MSVCRT_size_t *numberOfElements, const char *varname) |
| { |
| char* e; |
| MSVCRT_size_t sz; |
| |
| if (!MSVCRT_CHECK_PMT(buffer != NULL)) return MSVCRT_EINVAL; |
| if (!MSVCRT_CHECK_PMT(varname != NULL)) return MSVCRT_EINVAL; |
| |
| if (!(e = MSVCRT_getenv(varname))) return *MSVCRT__errno() = MSVCRT_EINVAL; |
| |
| sz = strlen(e) + 1; |
| if (!(*buffer = MSVCRT_malloc(sz))) |
| { |
| if (numberOfElements) *numberOfElements = 0; |
| return *MSVCRT__errno() = MSVCRT_ENOMEM; |
| } |
| strcpy(*buffer, e); |
| if (numberOfElements) *numberOfElements = sz; |
| return 0; |
| } |
| |
| /****************************************************************** |
| * _wdupenv_s (MSVCRT.@) |
| */ |
| int CDECL _wdupenv_s(MSVCRT_wchar_t **buffer, MSVCRT_size_t *numberOfElements, |
| const MSVCRT_wchar_t *varname) |
| { |
| MSVCRT_wchar_t* e; |
| MSVCRT_size_t sz; |
| |
| if (!MSVCRT_CHECK_PMT(buffer != NULL)) return MSVCRT_EINVAL; |
| if (!MSVCRT_CHECK_PMT(varname != NULL)) return MSVCRT_EINVAL; |
| |
| if (!(e = MSVCRT__wgetenv(varname))) return *MSVCRT__errno() = MSVCRT_EINVAL; |
| |
| sz = strlenW(e) + 1; |
| if (!(*buffer = MSVCRT_malloc(sz * sizeof(MSVCRT_wchar_t)))) |
| { |
| if (numberOfElements) *numberOfElements = 0; |
| return *MSVCRT__errno() = MSVCRT_ENOMEM; |
| } |
| strcpyW(*buffer, e); |
| if (numberOfElements) *numberOfElements = sz; |
| return 0; |
| } |
| |
| /****************************************************************** |
| * getenv_s (MSVCRT.@) |
| */ |
| int CDECL getenv_s(MSVCRT_size_t *pReturnValue, char* buffer, MSVCRT_size_t numberOfElements, const char *varname) |
| { |
| char* e; |
| |
| if (!MSVCRT_CHECK_PMT(pReturnValue != NULL)) return MSVCRT_EINVAL; |
| if (!MSVCRT_CHECK_PMT(!(buffer == NULL && numberOfElements > 0))) return MSVCRT_EINVAL; |
| if (!MSVCRT_CHECK_PMT(varname != NULL)) return MSVCRT_EINVAL; |
| |
| if (!(e = MSVCRT_getenv(varname))) |
| { |
| *pReturnValue = 0; |
| return *MSVCRT__errno() = MSVCRT_EINVAL; |
| } |
| *pReturnValue = strlen(e) + 1; |
| if (numberOfElements < *pReturnValue) |
| { |
| return *MSVCRT__errno() = MSVCRT_ERANGE; |
| } |
| strcpy(buffer, e); |
| return 0; |
| } |
| |
| /****************************************************************** |
| * _wgetenv_s (MSVCRT.@) |
| */ |
| int CDECL _wgetenv_s(MSVCRT_size_t *pReturnValue, MSVCRT_wchar_t *buffer, MSVCRT_size_t numberOfElements, |
| const MSVCRT_wchar_t *varname) |
| { |
| MSVCRT_wchar_t* e; |
| |
| if (!MSVCRT_CHECK_PMT(pReturnValue != NULL)) return MSVCRT_EINVAL; |
| if (!MSVCRT_CHECK_PMT(!(buffer == NULL && numberOfElements > 0))) return MSVCRT_EINVAL; |
| if (!MSVCRT_CHECK_PMT(varname != NULL)) return MSVCRT_EINVAL; |
| |
| if (!(e = MSVCRT__wgetenv(varname))) |
| { |
| *pReturnValue = 0; |
| return *MSVCRT__errno() = MSVCRT_EINVAL; |
| } |
| *pReturnValue = strlenW(e) + 1; |
| if (numberOfElements < *pReturnValue) |
| { |
| return *MSVCRT__errno() = MSVCRT_ERANGE; |
| } |
| strcpyW(buffer, e); |
| return 0; |
| } |