1 /*
2  * Unit test suite for dir functions
3  *
4  * Copyright 2006 CodeWeavers, Aric Stewart
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 <stdarg.h>
23 #include <stdlib.h>
24 #include <stdio.h>
25 #include <fcntl.h>
26 #include <direct.h>
27 #include <sys/stat.h>
28 #include <io.h>
29 #include <mbctype.h>
30 #include <windef.h>
31 #include <winbase.h>
32 #include <winnls.h>
33 #include <process.h>
34 #include <errno.h>
35 
36 static int (__cdecl *p_makepath_s)(char *, size_t, const char *, const char *, const char *, const char *);
37 static int (__cdecl *p_wmakepath_s)(wchar_t *, size_t, const wchar_t *,const wchar_t *, const wchar_t *, const wchar_t *);
38 static int (__cdecl *p_searchenv_s)(const char*, const char*, char*, size_t);
39 static int (__cdecl *p_wsearchenv_s)(const wchar_t*, const wchar_t*, wchar_t*, size_t);
40 
41 static void init(void)
42 {
43     HMODULE hmod = GetModuleHandleA("msvcrt.dll");
44 
45     p_makepath_s = (void *)GetProcAddress(hmod, "_makepath_s");
46     p_wmakepath_s = (void *)GetProcAddress(hmod, "_wmakepath_s");
47     p_searchenv_s = (void *)GetProcAddress(hmod, "_searchenv_s");
48     p_wsearchenv_s = (void *)GetProcAddress(hmod, "_wsearchenv_s");
49 }
50 
51 typedef struct
52 {
53     const char* buffer;
54     const char* drive;
55     const char* dir;
56     const char* file;
57     const char* ext;
58     const char* expected;
59 } makepath_case;
60 
61 #define USE_BUFF ((char*)~0ul)
62 static const makepath_case makepath_cases[] =
63 {
64     { NULL, NULL, NULL, NULL, NULL, "" }, /* 0 */
65     { NULL, "c", NULL, NULL, NULL, "c:" },
66     { NULL, "c:", NULL, NULL, NULL, "c:" },
67     { NULL, "c:\\", NULL, NULL, NULL, "c:" },
68     { NULL, NULL, "dir", NULL, NULL, "dir\\" },
69     { NULL, NULL, "dir\\", NULL, NULL, "dir\\" },
70     { NULL, NULL, "\\dir", NULL, NULL, "\\dir\\" },
71     { NULL, NULL, NULL, "file", NULL, "file" },
72     { NULL, NULL, NULL, "\\file", NULL, "\\file" },
73     { NULL, NULL, NULL, "file", NULL, "file" },
74     { NULL, NULL, NULL, NULL, "ext", ".ext" }, /* 10 */
75     { NULL, NULL, NULL, NULL, ".ext", ".ext" },
76     { "foo", NULL, NULL, NULL, NULL, "" },
77     { "foo", USE_BUFF, NULL, NULL, NULL, "f:" },
78     { "foo", NULL, USE_BUFF, NULL, NULL, "foo\\" },
79     { "foo", NULL, NULL, USE_BUFF, NULL, "foo" },
80     { "foo", NULL, USE_BUFF, "file", NULL, "foo\\file" },
81     { "foo", NULL, USE_BUFF, "file", "ext", "foo\\file.ext" },
82     { "foo", NULL, NULL, USE_BUFF, "ext", "foo.ext" },
83     /* remaining combinations of USE_BUFF crash native */
84     { NULL, "c", "dir", "file", "ext", "c:dir\\file.ext" },
85     { NULL, "c:", "dir", "file", "ext", "c:dir\\file.ext" }, /* 20 */
86     { NULL, "c:\\", "dir", "file", "ext", "c:dir\\file.ext" }
87 };
88 
89 static void test_makepath(void)
90 {
91     WCHAR driveW[MAX_PATH];
92     WCHAR dirW[MAX_PATH];
93     WCHAR fileW[MAX_PATH];
94     WCHAR extW[MAX_PATH];
95     WCHAR bufferW[MAX_PATH];
96     char buffer[MAX_PATH];
97 
98     unsigned int i, n;
99 
100     for (i = 0; i < sizeof(makepath_cases)/sizeof(makepath_cases[0]); ++i)
101     {
102         const makepath_case* p = &makepath_cases[i];
103 
104         memset(buffer, 'X', MAX_PATH);
105         if (p->buffer)
106             strcpy(buffer, p->buffer);
107 
108         /* Ascii */
109         _makepath(buffer,
110                   p->drive == USE_BUFF ? buffer : p->drive,
111                   p->dir == USE_BUFF ? buffer : p->dir,
112                   p->file == USE_BUFF? buffer : p->file,
113                   p->ext == USE_BUFF ? buffer : p->ext);
114 
115         buffer[MAX_PATH - 1] = '\0';
116         ok(!strcmp(p->expected, buffer), "got '%s' for case %d\n", buffer, i);
117 
118         /* Unicode */
119         if (p->drive != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->drive, -1, driveW, MAX_PATH);
120         if (p->dir != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->dir, -1, dirW, MAX_PATH);
121         if (p->file != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->file, -1, fileW, MAX_PATH);
122         if (p->ext != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->ext, -1, extW, MAX_PATH);
123 
124         memset(buffer, 0, MAX_PATH);
125         for (n = 0; n < MAX_PATH; ++n)
126             bufferW[n] = 'X';
127         if (p->buffer) MultiByteToWideChar( CP_ACP, 0, p->buffer, -1, bufferW, MAX_PATH);
128 
129         _wmakepath(bufferW,
130                    p->drive == USE_BUFF ? bufferW : p->drive ? driveW : NULL,
131                    p->dir == USE_BUFF ? bufferW : p->dir ? dirW : NULL,
132                    p->file == USE_BUFF? bufferW : p->file ? fileW : NULL,
133                    p->ext == USE_BUFF ? bufferW : p->ext ? extW : NULL);
134 
135         bufferW[MAX_PATH - 1] = '\0';
136         WideCharToMultiByte(CP_ACP, 0, bufferW, -1, buffer, MAX_PATH, NULL, NULL);
137         ok(!strcmp(p->expected, buffer), "got '%s' for unicode case %d\n", buffer, i);
138     }
139 }
140 
141 static const WCHAR expected0[] = {'\0','X','X','X','X','X','X','X','X','X','X','X','X'};
142 static const WCHAR expected1[] = {'\0','X','X','X','X','X','X','X','X','X','X','X','X'};
143 static const WCHAR expected2[] = {'\0',':','X','X','X','X','X','X','X','X','X','X','X'};
144 static const WCHAR expected3[] = {'\0',':','d','X','X','X','X','X','X','X','X','X','X'};
145 static const WCHAR expected4[] = {'\0',':','d','\\','X','X','X','X','X','X','X','X','X'};
146 static const WCHAR expected5[] = {'\0',':','d','\\','f','X','X','X','X','X','X','X','X'};
147 static const WCHAR expected6[] = {'\0',':','d','\\','f','i','X','X','X','X','X','X','X'};
148 static const WCHAR expected7[] = {'\0',':','d','\\','f','i','l','X','X','X','X','X','X'};
149 static const WCHAR expected8[] = {'\0',':','d','\\','f','i','l','e','X','X','X','X','X'};
150 static const WCHAR expected9[] = {'\0',':','d','\\','f','i','l','e','.','X','X','X','X'};
151 static const WCHAR expected10[] = {'\0',':','d','\\','f','i','l','e','.','e','X','X','X'};
152 static const WCHAR expected11[] = {'\0',':','d','\\','f','i','l','e','.','e','x','X','X'};
153 
154 static const WCHAR expected12[] = {'\0',':','X','X','X','X','X','X','X','X'};
155 static const WCHAR expected13[] = {'\0',':','d','X','X','X','X','X','X','X'};
156 static const WCHAR expected14[] = {'\0',':','d','i','X','X','X','X','X','X'};
157 static const WCHAR expected15[] = {'\0',':','d','i','r','X','X','X','X','X'};
158 static const WCHAR expected16[] = {'\0',':','d','i','r','\\','X','X','X','X'};
159 
160 static const WCHAR expected17[] = {'\0','o','o'};
161 static const WCHAR expected18[] = {'\0','o','o','\0','X'};
162 static const WCHAR expected19[] = {'\0','o','o','\0'};
163 static const WCHAR expected20[] = {'\0','o','o','\0','X','X','X','X','X'};
164 static const WCHAR expected21[] = {'\0','o','o','\\','f','i','l','X','X'};
165 static const WCHAR expected22[] = {'\0','o','o','\0','X','X','X','X','X','X','X','X','X'};
166 static const WCHAR expected23[] = {'\0','o','o','\\','f','i','l','X','X','X','X','X','X'};
167 static const WCHAR expected24[] = {'\0','o','o','\\','f','i','l','e','.','e','x','X','X'};
168 static const WCHAR expected25[] = {'\0','o','o','\0','X','X','X','X'};
169 static const WCHAR expected26[] = {'\0','o','o','.','e','x','X','X'};
170 
171 typedef struct
172 {
173     const char* buffer;
174     size_t length;
175     const char* drive;
176     const char* dir;
177     const char* file;
178     const char* ext;
179     const char* expected;
180     const WCHAR *expected_unicode;
181     size_t expected_length;
182 } makepath_s_case;
183 
184 static const makepath_s_case makepath_s_cases[] =
185 {
186     /* Behavior with directory parameter containing backslash. */
187     {NULL, 1, "c:", "d\\", "file", "ext", "\0XXXXXXXXXXXX", expected0, 13},
188     {NULL, 2, "c:", "d\\", "file", "ext", "\0XXXXXXXXXXXX", expected1, 13},
189     {NULL, 3, "c:", "d\\", "file", "ext", "\0:XXXXXXXXXXX", expected2, 13},
190     {NULL, 4, "c:", "d\\", "file", "ext", "\0:dXXXXXXXXXX", expected3, 13},
191     {NULL, 5, "c:", "d\\", "file", "ext", "\0:d\\XXXXXXXXX", expected4, 13},
192     {NULL, 6, "c:", "d\\", "file", "ext", "\0:d\\fXXXXXXXX", expected5, 13},
193     {NULL, 7, "c:", "d\\", "file", "ext", "\0:d\\fiXXXXXXX", expected6, 13},
194     {NULL, 8, "c:", "d\\", "file", "ext", "\0:d\\filXXXXXX", expected7, 13},
195     {NULL, 9, "c:", "d\\", "file", "ext", "\0:d\\fileXXXXX", expected8, 13},
196     {NULL, 10, "c:", "d\\", "file", "ext", "\0:d\\file.XXXX", expected9, 13},
197     {NULL, 11, "c:", "d\\", "file", "ext", "\0:d\\file.eXXX", expected10, 13},
198     {NULL, 12, "c:", "d\\", "file", "ext", "\0:d\\file.exXX", expected11, 13},
199     /* Behavior with directory parameter lacking backslash. */
200     {NULL, 3, "c:", "dir", "f", "ext", "\0:XXXXXXXX", expected12, 10},
201     {NULL, 4, "c:", "dir", "f", "ext", "\0:dXXXXXXX", expected13, 10},
202     {NULL, 5, "c:", "dir", "f", "ext", "\0:diXXXXXX", expected14, 10},
203     {NULL, 6, "c:", "dir", "f", "ext", "\0:dirXXXXX", expected15, 10},
204     {NULL, 7, "c:", "dir", "f", "ext", "\0:dir\\XXXX", expected16, 10},
205     /* Behavior with overlapped buffer. */
206     {"foo", 2, USE_BUFF, NULL, NULL, NULL, "\0oo", expected17, 3},
207     {"foo", 4, NULL, USE_BUFF, NULL, NULL, "\0oo\0X", expected18, 5},
208     {"foo", 3, NULL, NULL, USE_BUFF, NULL, "\0oo\0", expected19, 4},
209     {"foo", 4, NULL, USE_BUFF, "file", NULL, "\0oo\0XXXXX", expected20, 9},
210     {"foo", 8, NULL, USE_BUFF, "file", NULL, "\0oo\\filXX", expected21, 9},
211     {"foo", 4, NULL, USE_BUFF, "file", "ext", "\0oo\0XXXXXXXXX", expected22, 13},
212     {"foo", 8, NULL, USE_BUFF, "file", "ext", "\0oo\\filXXXXXX", expected23, 13},
213     {"foo", 12, NULL, USE_BUFF, "file", "ext", "\0oo\\file.exXX", expected24, 13},
214     {"foo", 4, NULL, NULL, USE_BUFF, "ext", "\0oo\0XXXX", expected25, 8},
215     {"foo", 7, NULL, NULL, USE_BUFF, "ext", "\0oo.exXX", expected26, 8},
216 };
217 
218 static void test_makepath_s(void)
219 {
220     WCHAR driveW[MAX_PATH];
221     WCHAR dirW[MAX_PATH];
222     WCHAR fileW[MAX_PATH];
223     WCHAR extW[MAX_PATH];
224     WCHAR bufferW[MAX_PATH];
225     char buffer[MAX_PATH];
226     int ret;
227     unsigned int i, n;
228 
229     if (!p_makepath_s || !p_wmakepath_s)
230     {
231         win_skip("Safe makepath functions are not available\n");
232         return;
233     }
234 
235     errno = EBADF;
236     ret = p_makepath_s(NULL, 0, NULL, NULL, NULL, NULL);
237     ok(ret == EINVAL, "Expected _makepath_s to return EINVAL, got %d\n", ret);
238     ok(errno == EINVAL, "Expected errno to be EINVAL, got %d\n", errno);
239 
240     errno = EBADF;
241     ret = p_makepath_s(buffer, 0, NULL, NULL, NULL, NULL);
242     ok(ret == EINVAL, "Expected _makepath_s to return EINVAL, got %d\n", ret);
243     ok(errno == EINVAL, "Expected errno to be EINVAL, got %d\n", errno);
244 
245     errno = EBADF;
246     ret = p_wmakepath_s(NULL, 0, NULL, NULL, NULL, NULL);
247     ok(ret == EINVAL, "Expected _wmakepath_s to return EINVAL, got %d\n", ret);
248     ok(errno == EINVAL, "Expected errno to be EINVAL, got %d\n", errno);
249 
250     errno = EBADF;
251     ret = p_wmakepath_s(bufferW, 0, NULL, NULL, NULL, NULL);
252     ok(ret == EINVAL, "Expected _wmakepath_s to return EINVAL, got %d\n", ret);
253     ok(errno == EINVAL, "Expected errno to be EINVAL, got %d\n", errno);
254 
255     /* Test with the normal _makepath cases. */
256     for (i = 0; i < sizeof(makepath_cases)/sizeof(makepath_cases[0]); i++)
257     {
258         const makepath_case *p = makepath_cases + i;
259 
260         memset(buffer, 'X', MAX_PATH);
261         if (p->buffer)
262             strcpy(buffer, p->buffer);
263 
264         /* Ascii */
265         ret = p_makepath_s(buffer, MAX_PATH,
266                            p->drive == USE_BUFF ? buffer : p->drive,
267                            p->dir == USE_BUFF ? buffer : p->dir,
268                            p->file == USE_BUFF? buffer : p->file,
269                            p->ext == USE_BUFF ? buffer : p->ext);
270         ok(ret == 0, "[%d] Expected _makepath_s to return 0, got %d\n", i, ret);
271 
272         buffer[MAX_PATH - 1] = '\0';
273         ok(!strcmp(p->expected, buffer), "got '%s' for case %d\n", buffer, i);
274 
275         /* Unicode */
276         if (p->drive != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->drive, -1, driveW, MAX_PATH);
277         if (p->dir != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->dir, -1, dirW, MAX_PATH);
278         if (p->file != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->file, -1, fileW, MAX_PATH);
279         if (p->ext != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->ext, -1, extW, MAX_PATH);
280 
281         memset(buffer, 0, MAX_PATH);
282         for (n = 0; n < MAX_PATH; ++n)
283             bufferW[n] = 'X';
284         if (p->buffer) MultiByteToWideChar( CP_ACP, 0, p->buffer, -1, bufferW, MAX_PATH);
285 
286         ret = p_wmakepath_s(bufferW, MAX_PATH,
287                             p->drive == USE_BUFF ? bufferW : p->drive ? driveW : NULL,
288                             p->dir == USE_BUFF ? bufferW : p->dir ? dirW : NULL,
289                             p->file == USE_BUFF? bufferW : p->file ? fileW : NULL,
290                             p->ext == USE_BUFF ? bufferW : p->ext ? extW : NULL);
291         ok(ret == 0, "[%d] Expected _wmakepath_s to return 0, got %d\n", i, ret);
292 
293         bufferW[MAX_PATH - 1] = '\0';
294         WideCharToMultiByte(CP_ACP, 0, bufferW, -1, buffer, MAX_PATH, NULL, NULL);
295         ok(!strcmp(p->expected, buffer), "got '%s' for unicode case %d\n", buffer, i);
296     }
297 
298     /* Try insufficient length cases. */
299     for (i = 0; i < sizeof(makepath_s_cases)/sizeof(makepath_s_cases[0]); i++)
300     {
301         const makepath_s_case *p = makepath_s_cases + i;
302 
303         memset(buffer, 'X', MAX_PATH);
304         if (p->buffer)
305             strcpy(buffer, p->buffer);
306 
307         /* Ascii */
308         errno = EBADF;
309         ret = p_makepath_s(buffer, p->length,
310                            p->drive == USE_BUFF ? buffer : p->drive,
311                            p->dir == USE_BUFF ? buffer : p->dir,
312                            p->file == USE_BUFF? buffer : p->file,
313                            p->ext == USE_BUFF ? buffer : p->ext);
314         ok(ret == ERANGE, "[%d] Expected _makepath_s to return ERANGE, got %d\n", i, ret);
315         ok(errno == ERANGE, "[%d] Expected errno to be ERANGE, got %d\n", i, errno);
316         ok(!memcmp(p->expected, buffer, p->expected_length), "unexpected output for case %d\n", i);
317 
318         /* Unicode */
319         if (p->drive != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->drive, -1, driveW, MAX_PATH);
320         if (p->dir != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->dir, -1, dirW, MAX_PATH);
321         if (p->file != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->file, -1, fileW, MAX_PATH);
322         if (p->ext != USE_BUFF) MultiByteToWideChar(CP_ACP, 0, p->ext, -1, extW, MAX_PATH);
323 
324         memset(buffer, 0, MAX_PATH);
325         for (n = 0; n < MAX_PATH; ++n)
326             bufferW[n] = 'X';
327         if (p->buffer) MultiByteToWideChar( CP_ACP, 0, p->buffer, -1, bufferW, MAX_PATH);
328 
329         errno = EBADF;
330         ret = p_wmakepath_s(bufferW, p->length,
331                             p->drive == USE_BUFF ? bufferW : p->drive ? driveW : NULL,
332                             p->dir == USE_BUFF ? bufferW : p->dir ? dirW : NULL,
333                             p->file == USE_BUFF? bufferW : p->file ? fileW : NULL,
334                             p->ext == USE_BUFF ? bufferW : p->ext ? extW : NULL);
335         ok(ret == ERANGE, "[%d] Expected _wmakepath_s to return ERANGE, got %d\n", i, ret);
336         ok(errno == ERANGE, "[%d] Expected errno to be ERANGE, got %d\n", i, errno);
337 
338         ok(!memcmp(p->expected_unicode, bufferW, p->expected_length * sizeof(WCHAR)), "unexpected output for case %d\n", i);
339     }
340 }
341 
342 static void test_fullpath(void)
343 {
344     char full[MAX_PATH];
345     char tmppath[MAX_PATH];
346     char prevpath[MAX_PATH];
347     char level1[MAX_PATH];
348     char level2[MAX_PATH];
349     char teststring[MAX_PATH];
350     char *freeme;
351     BOOL rc,free1,free2;
352 
353     free1=free2=TRUE;
354     GetCurrentDirectoryA(MAX_PATH, prevpath);
355     GetTempPathA(MAX_PATH,tmppath);
356     strcpy(level1,tmppath);
357     strcat(level1,"msvcrt-test\\");
358 
359     rc = CreateDirectoryA(level1,NULL);
360     if (!rc && GetLastError()==ERROR_ALREADY_EXISTS)
361         free1=FALSE;
362 
363     strcpy(level2,level1);
364     strcat(level2,"nextlevel\\");
365     rc = CreateDirectoryA(level2,NULL);
366     if (!rc && GetLastError()==ERROR_ALREADY_EXISTS)
367         free2=FALSE;
368     SetCurrentDirectoryA(level2);
369 
370     ok(_fullpath(full,"test", MAX_PATH)!=NULL,"_fullpath failed\n");
371     strcpy(teststring,level2);
372     strcat(teststring,"test");
373     ok(strcmp(full,teststring)==0,"Invalid Path returned %s\n",full);
374     ok(_fullpath(full,"\\test", MAX_PATH)!=NULL,"_fullpath failed\n");
375     memcpy(teststring,level2,3);
376     teststring[3]=0;
377     strcat(teststring,"test");
378     ok(strcmp(full,teststring)==0,"Invalid Path returned %s\n",full);
379     ok(_fullpath(full,"..\\test", MAX_PATH)!=NULL,"_fullpath failed\n");
380     strcpy(teststring,level1);
381     strcat(teststring,"test");
382     ok(strcmp(full,teststring)==0,"Invalid Path returned %s\n",full);
383     ok(_fullpath(full,"..\\test", 10)==NULL,"_fullpath failed to generate error\n");
384 
385     freeme = _fullpath(NULL,"test", 0);
386     ok(freeme!=NULL,"No path returned\n");
387     strcpy(teststring,level2);
388     strcat(teststring,"test");
389     ok(strcmp(freeme,teststring)==0,"Invalid Path returned %s\n",freeme);
390     free(freeme);
391 
392     SetCurrentDirectoryA(prevpath);
393     if (free2)
394         RemoveDirectoryA(level2);
395     if (free1)
396         RemoveDirectoryA(level1);
397 }
398 
399 static void test_splitpath(void)
400 {
401     const char* path = "c:\\\x83\x5c\x83\x74\x83\x67.bin";
402     char drive[3], dir[MAX_PATH], fname[MAX_PATH], ext[MAX_PATH];
403     int prev_cp = _getmbcp();
404 
405     /* SBCS codepage */
406     _setmbcp(1252);
407     _splitpath(path, drive, dir, fname, ext);
408     ok(!strcmp(drive, "c:"), "got %s\n", drive);
409     ok(!strcmp(dir, "\\\x83\x5c"), "got %s\n", dir);
410     ok(!strcmp(fname, "\x83\x74\x83\x67"), "got %s\n", fname);
411     ok(!strcmp(ext, ".bin"), "got %s\n", ext);
412 
413     /* MBCS (Japanese) codepage */
414     _setmbcp(932);
415     _splitpath(path, drive, dir, fname, ext);
416     ok(!strcmp(drive, "c:"), "got %s\n", drive);
417     ok(!strcmp(dir, "\\"), "got %s\n", dir);
418     ok(!strcmp(fname, "\x83\x5c\x83\x74\x83\x67"), "got %s\n", fname);
419     ok(!strcmp(ext, ".bin"), "got %s\n", ext);
420 
421     _setmbcp(prev_cp);
422 }
423 
424 static void test_searchenv(void)
425 {
426     const char *dirs[] = {
427         "\\search_env_test",
428         "\\search_env_test\\dir1",
429         "\\search_env_test\\dir2",
430         "\\search_env_test\\dir3longer"
431     };
432 
433     const char *files[] = {
434         "\\search_env_test\\dir1\\1.dat",
435         "\\search_env_test\\dir1\\2.dat",
436         "\\search_env_test\\dir2\\1.dat",
437         "\\search_env_test\\dir2\\3.dat",
438         "\\search_env_test\\dir3longer\\3.dat"
439     };
440 
441     const WCHAR env_w[] = {'T','E','S','T','_','P','A','T','H',0};
442     const WCHAR dat1_w[] = {'1','.','d','a','t',0};
443     const WCHAR dat3_w[] = {'3','.','d','a','t',0};
444 
445     char env1[4*MAX_PATH], env2[4*MAX_PATH], tmppath[MAX_PATH], path[2*MAX_PATH];
446     char result[MAX_PATH], exp[2*MAX_PATH];
447     WCHAR result_w[MAX_PATH];
448     int i, path_len;
449     FILE *tmp_file;
450 
451     if (getenv("TEST_PATH")) {
452         skip("TEST_PATH environment variable already set\n");
453         return;
454     }
455 
456     path_len = GetTempPathA(MAX_PATH, tmppath);
457     ok(path_len, "GetTempPath failed\n");
458     memcpy(path, tmppath, path_len);
459 
460     for(i=0; i<sizeof(dirs)/sizeof(*dirs); i++) {
461         strcpy(path+path_len, dirs[i]);
462 	ok(!mkdir(path), "mkdir failed (dir = %s)\n", path);
463     }
464 
465     for(i=0; i<sizeof(files)/sizeof(*files); i++) {
466         strcpy(path+path_len, files[i]);
467         tmp_file = fopen(path, "wb");
468 	ok(tmp_file != NULL, "fopen failed (file = %s)\n", path);
469         fclose(tmp_file);
470     }
471 
472     strcpy(env1, "TEST_PATH=");
473     strcpy(env2, "TEST_PATH=;");
474     for(i=1; i<sizeof(dirs)/sizeof(*dirs); i++) {
475         strcat(env1, tmppath);
476         strcat(env1, dirs[i]);
477         strcat(env1, ";");
478 
479         strcat(env2, tmppath);
480         strcat(env2, dirs[i]);
481         strcat(env2, ";;");
482     }
483 
484     if (!p_searchenv_s || !p_wsearchenv_s)
485         win_skip("searchenv_s or wsearchenv_s function is not available\n");
486 
487     putenv(env1);
488     memset(result, 'x', sizeof(result));
489     _searchenv("fail", "TEST_PATH", result);
490     ok(!result[0], "got %s, expected ''\n", result);
491 
492     if (p_searchenv_s) {
493         memset(result, 'x', sizeof(result));
494         i = p_searchenv_s("fail", "TEST_PATH", result, MAX_PATH);
495         ok(i == ENOENT, "searchenv_s returned %d\n", i);
496         ok(!result[0], "got %s, expected ''\n", result);
497     }
498 
499     memset(result, 'x', sizeof(result));
500     strcpy(exp, tmppath);
501     strcat(exp, files[0]);
502     _searchenv("1.dat", "TEST_PATH", result);
503     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
504 
505     if (p_searchenv_s) {
506         memset(result, 'x', sizeof(result));
507         i = p_searchenv_s("1.dat", "TEST_PATH", result, MAX_PATH);
508         ok(!i, "searchenv_s returned %d\n", i);
509         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
510     }
511 
512     memset(result_w, 'x', sizeof(result_w));
513     _wsearchenv(dat1_w, env_w, result_w);
514     WideCharToMultiByte(CP_ACP, 0, result_w, -1, result, MAX_PATH, NULL, NULL);
515     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
516 
517     if (p_wsearchenv_s) {
518         memset(result_w, 'x', sizeof(result_w));
519         i = p_wsearchenv_s(dat1_w, env_w, result_w, MAX_PATH);
520         ok(!i, "wsearchenv_s returned %d\n", i);
521         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
522     }
523 
524     memset(result, 'x', sizeof(result));
525     strcpy(exp, tmppath);
526     strcat(exp, files[3]);
527     _searchenv("3.dat", "TEST_PATH", result);
528     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
529 
530     if (p_searchenv_s) {
531         memset(result, 'x', sizeof(result));
532         i = p_searchenv_s("3.dat", "TEST_PATH", result, MAX_PATH);
533         ok(!i, "searchenv_s returned %d\n", i);
534         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
535     }
536 
537     memset(result_w, 'x', sizeof(result_w));
538     _wsearchenv(dat3_w, env_w, result_w);
539     WideCharToMultiByte(CP_ACP, 0, result_w, -1, result, MAX_PATH, NULL, NULL);
540     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
541 
542     if (p_wsearchenv_s) {
543         memset(result_w, 'x', sizeof(result_w));
544         i = p_wsearchenv_s(dat3_w, env_w, result_w, MAX_PATH);
545         ok(!i, "wsearchenv_s returned %d\n", i);
546         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
547     }
548 
549     putenv(env2);
550     memset(result, 'x', sizeof(result));
551     strcpy(exp, tmppath);
552     strcat(exp, files[0]);
553     _searchenv("1.dat", "TEST_PATH", result);
554     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
555 
556     if (p_searchenv_s) {
557         memset(result, 'x', sizeof(result));
558         i = p_searchenv_s("1.dat", "TEST_PATH", result, MAX_PATH);
559         ok(!i, "searchenv_s returned %d\n", i);
560         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
561     }
562 
563     memset(result_w, 'x', sizeof(result_w));
564     _wsearchenv(dat1_w, env_w, result_w);
565     WideCharToMultiByte(CP_ACP, 0, result_w, -1, result, MAX_PATH, NULL, NULL);
566     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
567 
568     if (p_wsearchenv_s) {
569         memset(result_w, 'x', sizeof(result_w));
570         i = p_wsearchenv_s(dat1_w, env_w, result_w, MAX_PATH);
571         ok(!i, "wsearchenv_s returned %d\n", i);
572         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
573     }
574 
575     memset(result, 'x', sizeof(result));
576     strcpy(exp, tmppath);
577     strcat(exp, files[3]);
578     _searchenv("3.dat", "TEST_PATH", result);
579     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
580 
581     if (p_searchenv_s) {
582         memset(result, 'x', sizeof(result));
583         i = p_searchenv_s("3.dat", "TEST_PATH", result, MAX_PATH);
584         ok(!i, "searchenv_s returned %d\n", i);
585         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
586     }
587 
588     memset(result_w, 'x', sizeof(result_w));
589     _wsearchenv(dat3_w, env_w, result_w);
590     WideCharToMultiByte(CP_ACP, 0, result_w, -1, result, MAX_PATH, NULL, NULL);
591     ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
592 
593     if (p_wsearchenv_s) {
594         memset(result_w, 'x', sizeof(result_w));
595         i = p_wsearchenv_s(dat3_w, env_w, result_w, MAX_PATH);
596         ok(!i, "wsearchenv_s returned %d\n", i);
597         ok(!strcmp(result, exp), "got %s, expected '%s'\n", result, exp);
598     }
599 
600     putenv("TEST_PATH=");
601 
602     for(i=sizeof(files)/sizeof(*files)-1; i>=0; i--) {
603         strcpy(path+path_len, files[i]);
604         ok(!remove(path), "remove failed (file = %s)\n", path);
605     }
606 
607     for(i=sizeof(dirs)/sizeof(*dirs)-1; i>=0; i--) {
608         strcpy(path+path_len, dirs[i]);
609         ok(!rmdir(path), "rmdir failed (dir = %s)\n", path);
610     }
611 }
612 
613 START_TEST(dir)
614 {
615     init();
616 
617     test_fullpath();
618     test_makepath();
619     test_makepath_s();
620     test_splitpath();
621     test_searchenv();
622 }
623