ReactOS 0.4.16-dev-942-g91fadeb
environ.c
Go to the documentation of this file.
1/*
2 * Unit tests for C library environment routines
3 *
4 * Copyright 2004 Mike Hearn <mh@codeweavers.com>
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 */
20
21#include "wine/test.h"
22#include <errno.h>
23#include <stdlib.h>
24#include <process.h>
25#include <winnls.h>
26
27#define DEFINE_EXPECT(func) \
28 static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
29
30#define SET_EXPECT(func) \
31 expect_ ## func = TRUE
32
33#define CHECK_EXPECT2(func) \
34 do { \
35 ok(expect_ ##func, "unexpected call " #func "\n"); \
36 called_ ## func = TRUE; \
37 }while(0)
38
39#define CHECK_EXPECT(func) \
40 do { \
41 CHECK_EXPECT2(func); \
42 expect_ ## func = FALSE; \
43 }while(0)
44
45#define CHECK_CALLED(func) \
46 do { \
47 ok(called_ ## func, "expected " #func "\n"); \
48 expect_ ## func = called_ ## func = FALSE; \
49 }while(0)
50
52
53static const char *a_very_long_env_string =
54 "LIBRARY_PATH="
55 "C:/Program Files/GLBasic/Compiler/platform/Win32/Bin/../lib/gcc/mingw32/3.4.2/;"
56 "C:/Program Files/GLBasic/Compiler/platform/Win32/Bin/../lib/gcc/;"
57 "/mingw/lib/gcc/mingw32/3.4.2/;"
58 "/usr/lib/gcc/mingw32/3.4.2/;"
59 "C:/Program Files/GLBasic/Compiler/platform/Win32/Bin/../lib/gcc/mingw32/3.4.2/../../../../mingw32/lib/mingw32/3.4.2/;"
60 "C:/Program Files/GLBasic/Compiler/platform/Win32/Bin/../lib/gcc/mingw32/3.4.2/../../../../mingw32/lib/;"
61 "/mingw/mingw32/lib/mingw32/3.4.2/;"
62 "/mingw/mingw32/lib/;"
63 "/mingw/lib/mingw32/3.4.2/;"
64 "/mingw/lib/;"
65 "C:/Program Files/GLBasic/Compiler/platform/Win32/Bin/../lib/gcc/mingw32/3.4.2/../../../mingw32/3.4.2/;"
66 "C:/Program Files/GLBasic/Compiler/platform/Win32/Bin/../lib/gcc/mingw32/3.4.2/../../../;"
67 "/mingw/lib/mingw32/3.4.2/;"
68 "/mingw/lib/;"
69 "/lib/mingw32/3.4.2/;"
70 "/lib/;"
71 "/usr/lib/mingw32/3.4.2/;"
72 "/usr/lib/";
73
74static char ***(__cdecl *p__p__environ)(void);
75static WCHAR ***(__cdecl *p__p__wenviron)(void);
77static wchar_t** (__cdecl *p_get_initial_wide_environment)(void);
78static errno_t (__cdecl *p_putenv_s)(const char*, const char*);
79static errno_t (__cdecl *p_wputenv_s)(const wchar_t*, const wchar_t*);
80static errno_t (__cdecl *p_getenv_s)(size_t*, char*, size_t, const char*);
81
82static char ***p_environ;
83static WCHAR ***p_wenviron;
84
85static void __cdecl test_invalid_parameter_handler(const wchar_t *expression,
86 const wchar_t *function, const wchar_t *file,
87 unsigned line, uintptr_t arg)
88{
90 ok(expression == NULL, "expression is not NULL\n");
91 ok(function == NULL, "function is not NULL\n");
92 ok(file == NULL, "file is not NULL\n");
93 ok(line == 0, "line = %u\n", line);
94 ok(arg == 0, "arg = %Ix\n", arg);
95}
96
97static BOOL init(void)
98{
99 HMODULE hmod = GetModuleHandleA( "ucrtbase.dll" );
100
101 p__p__environ = (void *)GetProcAddress( hmod, "__p__environ" );
102 p__p__wenviron = (void *)GetProcAddress( hmod, "__p__wenviron" );
103 p_get_initial_narrow_environment = (void *)GetProcAddress( hmod, "_get_initial_narrow_environment" );
104 p_get_initial_wide_environment = (void *)GetProcAddress( hmod, "_get_initial_wide_environment" );
105 p_putenv_s = (void *)GetProcAddress( hmod, "_putenv_s" );
106 p_wputenv_s = (void *)GetProcAddress( hmod, "_wputenv_s" );
107 p_getenv_s = (void *)GetProcAddress( hmod, "getenv_s" );
108
109 ok(p__p__environ != NULL, "Unexecped NULL pointer to environ\n" );
110 ok(p__p__wenviron != NULL, "Unexecped NULL pointer to environ\n" );
112 {
113 skip( "NULL pointers for environment\n" );
114 return FALSE;
115 }
118 return TRUE;
119}
120
121static unsigned env_get_entry_countA( char **env )
122{
123 unsigned count;
124
125 if (!env) return 0;
126 for (count = 0; env[count] != NULL; count++) {}
127 return count;
128}
129
130static void test_initial_environ( void )
131{
132 ok( p__p__environ() != NULL, "Unexpected NULL _environ[]\n" );
133 ok( *p__p__environ() != NULL, "Unexpected empty _environ[]\n" );
134 ok( p_get_initial_narrow_environment() != NULL, "Unexpected empty narrow initial environment\n" );
135 ok( p_get_initial_narrow_environment() == *p__p__environ(), "Expecting _environ[] to match initial narrow environment\n" );
136
137 ok( p__p__wenviron() != NULL, "Unexpected NULL _wenviron[]\n" );
138 ok( *p__p__wenviron() == NULL, "Unexpected non empty _wenviron[]\n" );
139 ok( p_get_initial_wide_environment() != NULL, "Unexpected empty wide initial environment\n" );
140 ok( p_get_initial_wide_environment() == *p__p__wenviron(), "Expecting _wenviron[] to match initial wide environment\n" );
141}
142
144{
145 char buf[256];
146 errno_t ret;
147 size_t len;
148 unsigned count;
149 char* first;
150 char* second;
151
152 ok( _putenv( "cat=" ) == 0, "_putenv failed on deletion of nonexistent environment variable\n" );
153 ok( _putenv( "cat=dog" ) == 0, "failed setting cat=dog\n" );
154 ok( strcmp( getenv( "cat" ), "dog" ) == 0, "getenv did not return 'dog'\n" );
155 if (p_getenv_s)
156 {
157 ret = p_getenv_s( &len, buf, sizeof(buf), "cat" );
158 ok( !ret, "getenv_s returned %d\n", ret );
159 ok( len == 4, "getenv_s returned length is %Id\n", len);
160 ok( !strcmp(buf, "dog"), "getenv_s did not return 'dog'\n" );
161 }
162 ok( _putenv("cat=") == 0, "failed deleting cat\n" );
163
164 ok( _putenv("=") == -1, "should not accept '=' as input\n" );
165 ok( _putenv("=dog") == -1, "should not accept '=dog' as input\n" );
166 ok( _putenv(a_very_long_env_string) == 0, "_putenv failed for long environment string\n" );
167
168 ok( getenv("nonexistent") == NULL, "getenv should fail with nonexistent var name\n" );
169
170 if (p_putenv_s)
171 {
173 ret = p_putenv_s( NULL, "dog" );
175 ok( ret == EINVAL, "_putenv_s returned %d\n", ret );
177 ret = p_putenv_s( "cat", NULL );
179 ok( ret == EINVAL, "_putenv_s returned %d\n", ret );
181 ret = p_putenv_s( "a=b", NULL );
183 ok( ret == EINVAL, "_putenv_s returned %d\n", ret );
184 ret = p_putenv_s( "cat", "a=b" );
185 ok( !ret, "_putenv_s returned %d\n", ret );
186 ret = p_putenv_s( "cat", "" );
187 ok( !ret, "_putenv_s returned %d\n", ret );
188 }
189
190 if (p_wputenv_s)
191 {
193 ret = p_wputenv_s( NULL, L"dog" );
195 ok( ret == EINVAL, "_wputenv_s returned %d\n", ret );
197 ret = p_wputenv_s( L"cat", NULL );
199 ok( ret == EINVAL, "_wputenv_s returned %d\n", ret );
201 ret = p_wputenv_s( L"a=b", NULL );
203 ok( ret == EINVAL, "_wputenv_s returned %d\n", ret );
204 ret = p_wputenv_s( L"cat", L"a=b" );
205 ok( !ret, "_wputenv_s returned %d\n", ret );
206 ret = p_wputenv_s( L"cat", L"" );
207 ok( !ret, "_wputenv_s returned %d\n", ret );
208 }
209
210 if (p_getenv_s)
211 {
212 buf[0] = 'x';
213 len = 1;
214 errno = 0xdeadbeef;
215 ret = p_getenv_s( &len, buf, sizeof(buf), "nonexistent" );
216 ok( !ret, "_getenv_s returned %d\n", ret );
217 ok( !len, "getenv_s returned length is %Id\n", len );
218 ok( !buf[0], "buf = %s\n", buf );
219 ok( errno == 0xdeadbeef, "errno = %d\n", errno );
220
221 buf[0] = 'x';
222 len = 1;
223 errno = 0xdeadbeef;
224 ret = p_getenv_s( &len, buf, sizeof(buf), NULL );
225 ok( !ret, "_getenv_s returned %d\n", ret );
226 ok( !len, "getenv_s returned length is %Id\n", len );
227 ok( !buf[0], "buf = %s\n", buf );
228 ok( errno == 0xdeadbeef, "errno = %d\n", errno );
229 }
230
231 /* test stability of _environ[] pointers */
232 ok( _putenv( "__winetest_cat=" ) == 0, "Couldn't reset env var\n" );
233 ok( _putenv( "__winetest_dog=" ) == 0, "Couldn't reset env var\n" );
235 ok( _putenv( "__winetest_cat=mew") == 0, "Couldn't set env var\n" );
236 ok( !strcmp( (*p_environ)[count], "__winetest_cat=mew"), "Unexpected env var value\n" );
237 first = (*p_environ)[count];
238 ok( getenv("__winetest_cat") == strchr( (*p_environ)[count], '=') + 1, "Expected getenv() to return pointer inside _environ[] entry\n" );
239 ok( _putenv( "__winetest_dog=bark" ) == 0, "Couldn't set env var\n" );
240 ok( !strcmp( (*p_environ)[count + 1], "__winetest_dog=bark" ), "Unexpected env var value\n" );
241 ok( getenv( "__winetest_dog" ) == strchr( (*p_environ)[count + 1], '=' ) + 1, "Expected getenv() to return pointer inside _environ[] entry\n" );
242 ok( first == (*p_environ)[count], "Expected stability of _environ[count] pointer\n" );
243 second = (*p_environ)[count + 1];
244 ok( count + 2 == env_get_entry_countA( *p_environ ), "Unexpected count\n" );
245
246 ok( _putenv( "__winetest_cat=purr" ) == 0, "Couldn't set env var\n" );
247 ok( !strcmp( (*p_environ)[count], "__winetest_cat=purr" ), "Unexpected env var value\n" );
248 ok( getenv( "__winetest_cat" ) == strchr( (*p_environ)[count], '=' ) + 1, "Expected getenv() to return pointer inside _environ[] entry\n" );
249 ok( second == (*p_environ)[count + 1], "Expected stability of _environ[count] pointer\n" );
250 ok( !strcmp( (*p_environ)[count + 1], "__winetest_dog=bark" ), "Couldn't get env var value\n" );
251 ok( getenv( "__winetest_dog" ) == strchr( (*p_environ)[count + 1], '=' ) + 1, "Expected getenv() to return pointer inside _environ[] entry\n" );
252 ok( count + 2 == env_get_entry_countA( *p_environ ), "Unexpected count\n" );
253 ok( _putenv( "__winetest_cat=" ) == 0, "Couldn't reset env vat\n" );
254 ok( second == (*p_environ)[count], "Expected _environ[count] to be second\n" );
255 ok( !strcmp( (*p_environ)[count], "__winetest_dog=bark" ), "Unexpected env var value\n" );
256 ok( count + 1 == env_get_entry_countA( *p_environ ), "Unexpected count\n" );
257 ok( _putenv( "__winetest_dog=" ) == 0, "Couldn't reset env var\n" );
258 ok( count == env_get_entry_countA( *p_environ ), "Unexpected count\n" );
259
260 /* in putenv, only changed variable is updated (no other reload of kernel info is done) */
261 ret = SetEnvironmentVariableA( "__winetest_cat", "meow" );
262 ok( ret, "SetEnvironmentVariableA failed: %lu\n", GetLastError() );
263 ok( _putenv( "__winetest_dog=bark" ) == 0, "Couldn't set env var\n" );
264 ok( getenv( "__winetest_cat" ) == NULL, "msvcrt env cache shouldn't have been updated\n" );
265 ok( _putenv( "__winetest_cat=" ) == 0, "Couldn't reset env var\n" );
266 ok( _putenv( "__winetest_dog=" ) == 0, "Couldn't reset env var\n" );
267
268 /* test setting unicode bits */
270 ret = WideCharToMultiByte( CP_ACP, 0, L"\u263a", -1, buf, ARRAY_SIZE(buf), 0, 0 );
271 ok( ret, "WideCharToMultiByte failed: %lu\n", GetLastError() );
272 ok( _wputenv( L"__winetest_cat=\u263a" ) == 0, "Couldn't set env var\n" );
273 ok( _wgetenv( L"__winetest_cat" ) && !wcscmp( _wgetenv( L"__winetest_cat" ), L"\u263a" ), "Couldn't retrieve env var\n" );
274 ok( getenv( "__winetest_cat" ) && !strcmp( getenv( "__winetest_cat" ), buf ), "Couldn't retrieve env var\n" );
275 ok( _wputenv( L"__winetest_cat=" ) == 0, "Couldn't reset env var\n" );
276
277 ret = WideCharToMultiByte( CP_ACP, 0, L"__winetest_\u263a", -1, buf, ARRAY_SIZE(buf), 0, 0 );
278 ok( ret, "WideCharToMultiByte failed: %lu\n", GetLastError() );
279 ok( _wputenv( L"__winetest_\u263a=bark" ) == 0, "Couldn't set env var\n" );
280 ok( _wgetenv( L"__winetest_\u263a" ) && !wcscmp( _wgetenv( L"__winetest_\u263a" ), L"bark"), "Couldn't retrieve env var\n" );
281 ok( getenv( buf ) && !strcmp( getenv( buf ), "bark"), "Couldn't retrieve env var %s\n", wine_dbgstr_a(buf) );
282 ok( _wputenv( L"__winetest_\u263a=" ) == 0, "Couldn't reset env var\n" );
283 ok( count == env_get_entry_countA( *p_environ ), "Unexpected modification of _environ[]\n" );
284}
285
286static void test_child_env(char** argv)
287{
288 STARTUPINFOA si = {sizeof(si)};
289 WCHAR *cur_env, *env, *p, *q;
291 char tmp[1024];
292 BOOL ret;
293 int len;
294
295 cur_env = GetEnvironmentStringsW();
296 ok( cur_env != NULL, "GetEnvironemntStrings failed\n" );
297
298 p = cur_env;
299 while (*p) p += wcslen( p ) + 1;
300 len = p - cur_env;
301 env = malloc( (len + 1024) * sizeof(*env) );
302 memcpy(env, cur_env, len * sizeof(*env) );
303 q = env + len;
304 FreeEnvironmentStringsW( cur_env );
305
306 wcscpy( q, L"__winetest_dog=bark" );
307 q += wcslen( L"__winetest_dog=bark" ) + 1;
308 wcscpy( q, L"__winetest_\u263a=\u03b2" );
309 q += wcslen( L"__winetest_\u263a=\u03b2" ) + 1;
310 *q = 0;
311
312 snprintf( tmp, sizeof(tmp), "%s %s create", argv[0], argv[1] );
314 ok( ret, "Couldn't create child process %s\n", tmp );
316 CloseHandle( pi.hProcess );
317 CloseHandle( pi.hThread );
318 free( env );
319}
320
322{
323 char **argv;
324 int argc;
325
326 if (!init()) return;
327
329 "Invalid parameter handler was already set\n" );
330
332 if (argc == 3 && !strcmp( argv[2], "create" ))
333 {
334 ok( getenv( "__winetest_dog" ) && !strcmp( getenv( "__winetest_dog" ), "bark" ),
335 "Couldn't find env var\n" );
336 ok( _wgetenv( L"__winetest_\u263a" ) && !wcscmp( _wgetenv( L"__winetest_\u263a" ), L"\u03b2" ),
337 "Couldn't find unicode env var\n" );
338 return;
339 }
340
344}
static int argc
Definition: ServiceArgs.c:12
static _invalid_parameter_handler invalid_parameter_handler
#define EINVAL
Definition: acclib.h:90
int strcmp(const char *String1, const char *String2)
Definition: utclib.c:469
char * strchr(const char *String, int ch)
Definition: utclib.c:501
#define __cdecl
Definition: accygwin.h:79
#define ok(value,...)
Definition: atltest.h:57
#define skip(...)
Definition: atltest.h:64
#define START_TEST(x)
Definition: atltest.h:75
#define ARRAY_SIZE(A)
Definition: main.h:20
wcscpy
static LPCWSTR LPCWSTR LPCWSTR env
Definition: db.cpp:170
#define free
Definition: debug_ros.c:5
#define malloc
Definition: debug_ros.c:4
#define NULL
Definition: types.h:112
#define TRUE
Definition: types.h:120
#define FALSE
Definition: types.h:117
#define CloseHandle
Definition: compat.h:739
#define CP_ACP
Definition: compat.h:109
#define GetProcAddress(x, y)
Definition: compat.h:753
#define WideCharToMultiByte
Definition: compat.h:111
BOOL WINAPI FreeEnvironmentStringsW(IN LPWSTR EnvironmentStrings)
Definition: environ.c:389
BOOL WINAPI DECLSPEC_HOTPATCH SetEnvironmentVariableA(IN LPCSTR lpName, IN LPCSTR lpValue)
Definition: environ.c:218
HMODULE WINAPI DECLSPEC_HOTPATCH GetModuleHandleA(LPCSTR lpModuleName)
Definition: loader.c:812
BOOL WINAPI DECLSPEC_HOTPATCH CreateProcessA(LPCSTR lpApplicationName, LPSTR lpCommandLine, LPSECURITY_ATTRIBUTES lpProcessAttributes, LPSECURITY_ATTRIBUTES lpThreadAttributes, BOOL bInheritHandles, DWORD dwCreationFlags, LPVOID lpEnvironment, LPCSTR lpCurrentDirectory, LPSTARTUPINFOA lpStartupInfo, LPPROCESS_INFORMATION lpProcessInformation)
Definition: proc.c:4747
unsigned int BOOL
Definition: ntddk_ex.h:94
GLuint GLuint GLsizei count
Definition: gl.h:1545
GLdouble GLdouble GLdouble GLdouble q
Definition: gl.h:2063
GLenum GLuint GLenum GLsizei const GLchar * buf
Definition: glext.h:7751
const GLint * first
Definition: glext.h:5794
GLfloat GLfloat p
Definition: glext.h:8902
GLenum GLsizei len
Definition: glext.h:6722
_invalid_parameter_handler __cdecl _set_invalid_parameter_handler(_In_opt_ _invalid_parameter_handler _Handler)
_Check_return_ _CRTIMP int __cdecl _wputenv(_In_z_ const wchar_t *_EnvString)
_Check_return_ _CRTIMP wchar_t *__cdecl _wgetenv(_In_z_ const wchar_t *_VarName)
#define environ
Definition: stdlib.h:1348
_Check_return_ _CRTIMP int __cdecl _putenv(_In_z_ const char *_EnvString)
_Check_return_ char *__cdecl getenv(_In_z_ const char *_VarName)
_CRTIMP size_t __cdecl wcslen(_In_z_ const wchar_t *_Str)
unsigned int uintptr_t
Definition: intrin.h:47
#define memcpy(s1, s2, n)
Definition: mkisofs.h:878
static PEXPLICIT_ACCESSW *static HMODULE hmod
Definition: security.c:143
static const char *static const wchar_t *static char const char *static char *** p_environ
Definition: environ.c:63
static void init(void)
Definition: environ.c:66
static const char *static const wchar_t *static char size_t
Definition: environ.c:61
static WCHAR ***__cdecl * p__p__wenviron(void)
static unsigned env_get_entry_countA(char **env)
Definition: environ.c:92
static char ***__cdecl * p__p__environ(void)
static void test_environment_manipulation(void)
Definition: environ.c:263
static const char * a_very_long_env_string
Definition: environ.c:32
static void test_child_env(char **argv)
Definition: environ.c:394
static WCHAR *** p_wenviron
Definition: environ.c:64
static refpint_t pi[]
Definition: server.c:96
#define SET_EXPECT(func)
Definition: environ.c:30
static void test_initial_environ(void)
Definition: environ.c:130
#define CHECK_EXPECT(func)
Definition: environ.c:39
#define DEFINE_EXPECT(func)
Definition: environ.c:27
static wchar_t **__cdecl * p_get_initial_wide_environment(void)
static char **__cdecl * p_get_initial_narrow_environment(void)
#define CHECK_CALLED(func)
Definition: environ.c:45
static void test_invalid_parameter_handler(void)
Definition: misc.c:384
#define argv
Definition: mplay32.c:18
#define L(x)
Definition: ntvdm.h:50
#define errno
Definition: errno.h:18
_Check_return_ _CRTIMP int __cdecl wcscmp(_In_z_ const wchar_t *_Str1, _In_z_ const wchar_t *_Str2)
const char int int int static __inline const char * wine_dbgstr_a(const char *s)
Definition: debug.h:187
int winetest_get_mainargs(char ***pargv)
void winetest_wait_child_process(HANDLE process)
Definition: fci.c:127
Definition: parser.c:49
int errno_t
Definition: corecrt.h:615
int ret
DWORD WINAPI GetLastError(void)
Definition: except.c:1042
#define CREATE_UNICODE_ENVIRONMENT
Definition: winbase.h:189
LPWSTR WINAPI GetEnvironmentStringsW(void)
Definition: environ.c:344
#define snprintf
Definition: wintirpc.h:48
__wchar_t WCHAR
Definition: xmlstorage.h:180