xref: /reactos/dll/win32/kernel32/client/file/find.c (revision 0b366ea1)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         ReactOS system libraries
4  * FILE:            dll/win32/kernel32/client/file/find.c
5  * PURPOSE:         Find functions
6  * PROGRAMMERS:     Ariadne (ariadne@xs4all.nl)
7  *                  Pierre Schweitzer (pierre.schweitzer@reactos.org)
8  *                  Hermes Belusca-Maito
9  */
10 
11 /* INCLUDES *******************************************************************/
12 
13 #include <k32.h>
14 #include <ntstrsafe.h>
15 
16 #define NDEBUG
17 #include <debug.h>
18 DEBUG_CHANNEL(kernel32file);
19 
20 
21 /* TYPES **********************************************************************/
22 
23 #define FIND_DATA_SIZE      0x4000
24 #define FIND_DEVICE_HANDLE  ((HANDLE)0x1)
25 
26 typedef enum _FIND_DATA_TYPE
27 {
28     FindFile   = 1,
29     FindStream = 2
30 } FIND_DATA_TYPE;
31 
32 /*
33  * FILE_FULL_DIR_INFORMATION and FILE_BOTH_DIR_INFORMATION structures layout.
34  *
35  *
36  *  struct FILE_FULL_DIR_INFORMATION   |   struct FILE_BOTH_DIR_INFORMATION
37  * ------------------------------------+---------------------------------------
38  *     ULONG NextEntryOffset;          |       ULONG NextEntryOffset;
39  *     ULONG FileIndex;                |       ULONG FileIndex;
40  *     LARGE_INTEGER CreationTime;     |       LARGE_INTEGER CreationTime;
41  *     LARGE_INTEGER LastAccessTime;   |       LARGE_INTEGER LastAccessTime;
42  *     LARGE_INTEGER LastWriteTime;    |       LARGE_INTEGER LastWriteTime;
43  *     LARGE_INTEGER ChangeTime;       |       LARGE_INTEGER ChangeTime;
44  *     LARGE_INTEGER EndOfFile;        |       LARGE_INTEGER EndOfFile;
45  *     LARGE_INTEGER AllocationSize;   |       LARGE_INTEGER AllocationSize;
46  *     ULONG FileAttributes;           |       ULONG FileAttributes;
47  *     ULONG FileNameLength;           |       ULONG FileNameLength;
48  *     ULONG EaSize;                   |       ULONG EaSize;
49  * ------------------------------------+---------------------------------------
50  *     WCHAR FileName[1];              |       CCHAR ShortNameLength;
51  *                                     |       WCHAR ShortName[12];
52  *                                     |       WCHAR FileName[1];
53  *
54  * Therefore we can use pointers to FILE_FULL_DIR_INFORMATION when one doesn't
55  * want to refer to the ShortName* fields and FileName (useful for implementing
56  * the FindExInfoBasic functionality for FindFirstFileEx), however a cast to
57  * FILE_BOTH_DIR_INFORMATION is required when one wants to use FileName and
58  * ShortName* fields (needed for the FindExInfoStandard functionality).
59  *
60  */
61 typedef union _DIR_INFORMATION
62 {
63     PVOID DirInfo;
64     PFILE_FULL_DIR_INFORMATION FullDirInfo;
65     PFILE_BOTH_DIR_INFORMATION BothDirInfo;
66 } DIR_INFORMATION;
67 
68 typedef struct _FIND_FILE_DATA
69 {
70     HANDLE Handle;
71     FINDEX_INFO_LEVELS InfoLevel;
72     FINDEX_SEARCH_OPS SearchOp;
73 
74     /*
75      * For handling STATUS_BUFFER_OVERFLOW errors emitted by
76      * NtQueryDirectoryFile in the FindNextFile function.
77      */
78     BOOLEAN HasMoreData;
79 
80     /*
81      * "Pointer" to the next file info structure in the buffer.
82      * The type is defined by the 'InfoLevel' parameter.
83      */
84     DIR_INFORMATION NextDirInfo;
85 
86     BYTE Buffer[FIND_DATA_SIZE];
87 } FIND_FILE_DATA, *PFIND_FILE_DATA;
88 
89 typedef struct _FIND_STREAM_DATA
90 {
91     STREAM_INFO_LEVELS InfoLevel;
92     PFILE_STREAM_INFORMATION FileStreamInfo;
93     PFILE_STREAM_INFORMATION CurrentInfo;
94 } FIND_STREAM_DATA, *PFIND_STREAM_DATA;
95 
96 typedef struct _FIND_DATA_HANDLE
97 {
98     FIND_DATA_TYPE Type;
99     RTL_CRITICAL_SECTION Lock;
100 
101     /*
102      * Pointer to the following finding data, located at
103      * (this + 1). The type is defined by the 'Type' parameter.
104      */
105     union
106     {
107         PFIND_FILE_DATA FindFileData;
108         PFIND_STREAM_DATA FindStreamData;
109     } u;
110 
111 } FIND_DATA_HANDLE, *PFIND_DATA_HANDLE;
112 
113 
114 /* PRIVATE FUNCTIONS **********************************************************/
115 
116 static VOID
117 CopyDeviceFindData(OUT LPWIN32_FIND_DATAW lpFindFileData,
118                    IN LPCWSTR lpFileName,
119                    IN ULONG DeviceNameInfo)
120 {
121     LPCWSTR DeviceName;
122     SIZE_T Length;
123 
124     _SEH2_TRY
125     {
126         /* DeviceNameInfo == { USHORT Offset; USHORT Length } */
127         Length     =  (SIZE_T)(DeviceNameInfo & 0xFFFF);
128         DeviceName = (LPCWSTR)((ULONG_PTR)lpFileName + ((DeviceNameInfo >> 16) & 0xFFFF));
129 
130         /* Return the data */
131         RtlZeroMemory(lpFindFileData, sizeof(*lpFindFileData));
132         lpFindFileData->dwFileAttributes = FILE_ATTRIBUTE_ARCHIVE;
133         RtlStringCbCopyNW(lpFindFileData->cFileName,
134                           sizeof(lpFindFileData->cFileName),
135                           DeviceName, Length);
136     }
137     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
138     {
139     }
140     _SEH2_END;
141 
142     return;
143 }
144 
145 static VOID
146 CopyFindData(OUT LPWIN32_FIND_DATAW lpFindFileData,
147              IN FINDEX_INFO_LEVELS fInfoLevelId,
148              IN DIR_INFORMATION DirInfo)
149 {
150 #define ULARGE_INTEGER_2_FILETIME(ft, ul) \
151 do { \
152     (ft).dwHighDateTime = (ul).u.HighPart; \
153     (ft).dwLowDateTime  = (ul).u.LowPart ; \
154 } while(0)
155 
156     _SEH2_TRY
157     {
158         RtlZeroMemory(lpFindFileData, sizeof(*lpFindFileData));
159 
160         lpFindFileData->dwFileAttributes = DirInfo.FullDirInfo->FileAttributes;
161 
162         ULARGE_INTEGER_2_FILETIME(lpFindFileData->ftCreationTime, DirInfo.FullDirInfo->CreationTime);
163         ULARGE_INTEGER_2_FILETIME(lpFindFileData->ftLastAccessTime, DirInfo.FullDirInfo->LastAccessTime);
164         ULARGE_INTEGER_2_FILETIME(lpFindFileData->ftLastWriteTime, DirInfo.FullDirInfo->LastWriteTime);
165 
166         lpFindFileData->nFileSizeHigh = DirInfo.FullDirInfo->EndOfFile.u.HighPart;
167         lpFindFileData->nFileSizeLow = DirInfo.FullDirInfo->EndOfFile.u.LowPart;
168 
169         /* dwReserved0 contains the NTFS reparse point tag, if any. */
170         if (DirInfo.FullDirInfo->FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT)
171             lpFindFileData->dwReserved0 = DirInfo.FullDirInfo->EaSize;
172         else
173             lpFindFileData->dwReserved0 = 0;
174 
175         /* Unused dwReserved1 field */
176         lpFindFileData->dwReserved1 = 0;
177 
178         if (fInfoLevelId == FindExInfoStandard)
179         {
180             RtlStringCbCopyNW(lpFindFileData->cFileName,
181                               sizeof(lpFindFileData->cFileName),
182                               DirInfo.BothDirInfo->FileName,
183                               DirInfo.BothDirInfo->FileNameLength);
184 
185             RtlStringCbCopyNW(lpFindFileData->cAlternateFileName,
186                               sizeof(lpFindFileData->cAlternateFileName),
187                               DirInfo.BothDirInfo->ShortName,
188                               DirInfo.BothDirInfo->ShortNameLength);
189         }
190         else if (fInfoLevelId == FindExInfoBasic)
191         {
192             RtlStringCbCopyNW(lpFindFileData->cFileName,
193                               sizeof(lpFindFileData->cFileName),
194                               DirInfo.FullDirInfo->FileName,
195                               DirInfo.FullDirInfo->FileNameLength);
196 
197             lpFindFileData->cAlternateFileName[0] = UNICODE_NULL;
198         }
199         else
200         {
201             /* Invalid InfoLevelId */
202             ASSERT(FALSE);
203         }
204     }
205     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
206     {
207     }
208     _SEH2_END;
209 
210     return;
211 }
212 
213 static VOID
214 CopyStreamData(IN OUT PFIND_STREAM_DATA FindStreamData,
215                OUT PWIN32_FIND_STREAM_DATA lpFindStreamData)
216 {
217     _SEH2_TRY
218     {
219         ASSERT(FindStreamData->CurrentInfo);
220 
221         switch (FindStreamData->InfoLevel)
222         {
223             case FindStreamInfoStandard:
224             {
225                 ULONG StreamNameLen = min(FindStreamData->CurrentInfo->StreamNameLength,
226                                           sizeof(lpFindStreamData->cStreamName) - sizeof(WCHAR));
227 
228                 RtlZeroMemory(lpFindStreamData, sizeof(*lpFindStreamData));
229 
230                 lpFindStreamData->StreamSize.QuadPart = FindStreamData->CurrentInfo->StreamSize.QuadPart;
231                 RtlCopyMemory(lpFindStreamData->cStreamName,
232                               FindStreamData->CurrentInfo->StreamName,
233                               StreamNameLen);
234                 lpFindStreamData->cStreamName[StreamNameLen / sizeof(WCHAR)] = UNICODE_NULL;
235 
236                 break;
237             }
238 
239             default:
240             {
241                 /* Invalid InfoLevel */
242                 ASSERT(FALSE);
243                 break;
244             }
245         }
246     }
247     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
248     {
249     }
250     _SEH2_END;
251 
252     return;
253 }
254 
255 
256 /* PUBLIC FUNCTIONS ***********************************************************/
257 
258 /*
259  * @implemented
260  */
261 HANDLE
262 WINAPI
263 FindFirstFileA(IN LPCSTR lpFileName,
264                OUT LPWIN32_FIND_DATAA lpFindFileData)
265 {
266     HANDLE hSearch;
267     NTSTATUS Status;
268     ANSI_STRING Ansi;
269     UNICODE_STRING UTF8;
270     PUNICODE_STRING lpFileNameW;
271     WIN32_FIND_DATAW FindFileDataW;
272 
273     lpFileNameW = Basep8BitStringToStaticUnicodeString(lpFileName);
274     if (!lpFileNameW) return INVALID_HANDLE_VALUE;
275 
276     hSearch = FindFirstFileExW(lpFileNameW->Buffer,
277                                FindExInfoStandard,
278                                &FindFileDataW,
279                                FindExSearchNameMatch,
280                                NULL, 0);
281     if (hSearch == INVALID_HANDLE_VALUE) return INVALID_HANDLE_VALUE;
282 
283     RtlCopyMemory(lpFindFileData,
284                   &FindFileDataW,
285                   FIELD_OFFSET(WIN32_FIND_DATAA, cFileName));
286 
287     RtlInitUnicodeString(&UTF8, FindFileDataW.cFileName);
288     Ansi.Buffer = lpFindFileData->cFileName;
289     Ansi.Length = 0;
290     Ansi.MaximumLength = sizeof(lpFindFileData->cFileName);
291     Status = BasepUnicodeStringTo8BitString(&Ansi, &UTF8, FALSE);
292     if (!NT_SUCCESS(Status))
293     {
294         FindClose(hSearch);
295         BaseSetLastNTError(Status);
296         return INVALID_HANDLE_VALUE;
297     }
298 
299     RtlInitUnicodeString(&UTF8, FindFileDataW.cAlternateFileName);
300     Ansi.Buffer = lpFindFileData->cAlternateFileName;
301     Ansi.Length = 0;
302     Ansi.MaximumLength = sizeof(lpFindFileData->cAlternateFileName);
303     Status = BasepUnicodeStringTo8BitString(&Ansi, &UTF8, FALSE);
304     if (!NT_SUCCESS(Status))
305     {
306         FindClose(hSearch);
307         BaseSetLastNTError(Status);
308         return INVALID_HANDLE_VALUE;
309     }
310 
311     return hSearch;
312 }
313 
314 
315 /*
316  * @implemented
317  */
318 HANDLE
319 WINAPI
320 FindFirstFileW(IN LPCWSTR lpFileName,
321                OUT LPWIN32_FIND_DATAW lpFindFileData)
322 {
323     return FindFirstFileExW(lpFileName,
324                             FindExInfoStandard,
325                             lpFindFileData,
326                             FindExSearchNameMatch,
327                             NULL, 0);
328 }
329 
330 
331 /*
332  * @implemented
333  */
334 BOOL
335 WINAPI
336 FindNextFileA(IN HANDLE hFindFile,
337               OUT LPWIN32_FIND_DATAA lpFindFileData)
338 {
339     NTSTATUS Status;
340     ANSI_STRING Ansi;
341     UNICODE_STRING UTF8;
342     WIN32_FIND_DATAW FindFileDataW;
343 
344     if (!FindNextFileW(hFindFile, &FindFileDataW))
345         return FALSE;
346 
347     RtlCopyMemory(lpFindFileData,
348                   &FindFileDataW,
349                   FIELD_OFFSET(WIN32_FIND_DATAA, cFileName));
350 
351     RtlInitUnicodeString(&UTF8, FindFileDataW.cFileName);
352     Ansi.Buffer = lpFindFileData->cFileName;
353     Ansi.Length = 0;
354     Ansi.MaximumLength = sizeof(lpFindFileData->cFileName);
355     Status = BasepUnicodeStringTo8BitString(&Ansi, &UTF8, FALSE);
356     if (!NT_SUCCESS(Status))
357     {
358         BaseSetLastNTError(Status);
359         return FALSE;
360     }
361 
362     RtlInitUnicodeString(&UTF8, FindFileDataW.cAlternateFileName);
363     Ansi.Buffer = lpFindFileData->cAlternateFileName;
364     Ansi.Length = 0;
365     Ansi.MaximumLength = sizeof(lpFindFileData->cAlternateFileName);
366     Status = BasepUnicodeStringTo8BitString(&Ansi, &UTF8, FALSE);
367     if (!NT_SUCCESS(Status))
368     {
369         BaseSetLastNTError(Status);
370         return FALSE;
371     }
372 
373     return TRUE;
374 }
375 
376 
377 /*
378  * @implemented
379  */
380 BOOL
381 WINAPI
382 FindNextFileW(IN HANDLE hFindFile,
383               OUT LPWIN32_FIND_DATAW lpFindFileData)
384 {
385     NTSTATUS Status = STATUS_SUCCESS;
386     DIR_INFORMATION FoundFile = {NULL};
387 
388     TRACE("FindNextFileW(%p, 0x%p)\n", hFindFile, lpFindFileData);
389 
390     if (hFindFile != FIND_DEVICE_HANDLE)
391     {
392         PFIND_DATA_HANDLE FindDataHandle = (PFIND_DATA_HANDLE)hFindFile;
393         PFIND_FILE_DATA FindFileData;
394         FINDEX_INFO_LEVELS InfoLevel;
395         IO_STATUS_BLOCK IoStatusBlock;
396         DIR_INFORMATION DirInfo = {NULL}, NextDirInfo = {NULL};
397 
398         if (hFindFile == NULL || hFindFile == INVALID_HANDLE_VALUE ||
399             FindDataHandle->Type != FindFile)
400         {
401             SetLastError(ERROR_INVALID_HANDLE);
402             return FALSE;
403         }
404 
405         RtlEnterCriticalSection(&FindDataHandle->Lock);
406 
407         FindFileData = FindDataHandle->u.FindFileData;
408         InfoLevel = FindFileData->InfoLevel;
409 
410         do
411         {
412             if (FindFileData->NextDirInfo.DirInfo == NULL)
413             {
414                 Status = NtQueryDirectoryFile(FindFileData->Handle,
415                                               NULL, NULL, NULL,
416                                               &IoStatusBlock,
417                                               &FindFileData->Buffer,
418                                               sizeof(FindFileData->Buffer),
419                                               (InfoLevel == FindExInfoStandard
420                                                           ? FileBothDirectoryInformation
421                                                           : FileFullDirectoryInformation),
422                                               FALSE,
423                                               NULL, /* Use the file pattern from the first call */
424                                               FALSE);
425                 if (Status == STATUS_BUFFER_OVERFLOW)
426                 {
427                     FindFileData->HasMoreData = TRUE;
428                     Status = STATUS_SUCCESS;
429                 }
430                 else
431                 {
432                     if (!NT_SUCCESS(Status)) break;
433                     FindFileData->HasMoreData = FALSE;
434                 }
435 
436                 FindFileData->NextDirInfo.DirInfo = &FindFileData->Buffer;
437             }
438 
439             DirInfo = FindFileData->NextDirInfo;
440 
441             if (DirInfo.FullDirInfo->NextEntryOffset != 0)
442             {
443                 ULONG_PTR BufferEnd = (ULONG_PTR)&FindFileData->Buffer + sizeof(FindFileData->Buffer);
444                 PWSTR pFileName;
445 
446                 NextDirInfo.DirInfo = FindFileData->NextDirInfo.DirInfo =
447                     (PVOID)((ULONG_PTR)DirInfo.DirInfo + DirInfo.FullDirInfo->NextEntryOffset);
448 
449                 pFileName = (InfoLevel == FindExInfoStandard
450                                         ? NextDirInfo.BothDirInfo->FileName
451                                         : NextDirInfo.FullDirInfo->FileName);
452 
453                 /* Be paranoid and make sure that the next entry is completely there */
454                 if (BufferEnd < (ULONG_PTR)NextDirInfo.DirInfo ||
455                     BufferEnd < (ULONG_PTR)&NextDirInfo.FullDirInfo->FileNameLength + sizeof(NextDirInfo.FullDirInfo->FileNameLength) ||
456                     BufferEnd <= (ULONG_PTR)((ULONG_PTR)pFileName + NextDirInfo.FullDirInfo->FileNameLength))
457                 {
458                     FindFileData->NextDirInfo.DirInfo = NULL;
459                 }
460             }
461             else
462             {
463                 FindFileData->NextDirInfo.DirInfo = NULL;
464             }
465 
466             if ((FindFileData->SearchOp != FindExSearchLimitToDirectories) ||
467                 (DirInfo.FullDirInfo->FileAttributes & FILE_ATTRIBUTE_DIRECTORY))
468             {
469                 FoundFile = DirInfo;
470             }
471         } while ( FoundFile.DirInfo == NULL && (FindFileData->NextDirInfo.DirInfo || FindFileData->HasMoreData) );
472 
473         if (FoundFile.DirInfo != NULL)
474         {
475             /* Return the information */
476             CopyFindData(lpFindFileData, InfoLevel, FoundFile);
477         }
478 
479         RtlLeaveCriticalSection(&FindDataHandle->Lock);
480     }
481 
482     if (!NT_SUCCESS(Status))
483     {
484         BaseSetLastNTError(Status);
485         return FALSE;
486     }
487     else if (FoundFile.DirInfo == NULL)
488     {
489         SetLastError(ERROR_NO_MORE_FILES);
490         return FALSE;
491     }
492 
493     return TRUE;
494 }
495 
496 
497 /*
498  * @implemented
499  */
500 BOOL
501 WINAPI
502 FindClose(HANDLE hFindFile)
503 {
504     TRACE("FindClose(hFindFile %p)\n", hFindFile);
505 
506     if (hFindFile == FIND_DEVICE_HANDLE)
507         return TRUE;
508 
509     if (!hFindFile || hFindFile == INVALID_HANDLE_VALUE)
510     {
511         SetLastError(ERROR_INVALID_HANDLE);
512         return FALSE;
513     }
514 
515     /* Protect with SEH against closing attempts on invalid handles. */
516     _SEH2_TRY
517     {
518         PFIND_DATA_HANDLE FindDataHandle = (PFIND_DATA_HANDLE)hFindFile;
519 
520         switch (FindDataHandle->Type)
521         {
522             case FindFile:
523             {
524                 RtlEnterCriticalSection(&FindDataHandle->Lock);
525                 NtClose(FindDataHandle->u.FindFileData->Handle);
526                 RtlLeaveCriticalSection(&FindDataHandle->Lock);
527                 RtlDeleteCriticalSection(&FindDataHandle->Lock);
528                 break;
529             }
530 
531             case FindStream:
532             {
533                 RtlEnterCriticalSection(&FindDataHandle->Lock);
534                 if (FindDataHandle->u.FindStreamData->FileStreamInfo != NULL)
535                 {
536                     RtlFreeHeap(RtlGetProcessHeap(), 0,
537                                 FindDataHandle->u.FindStreamData->FileStreamInfo);
538                 }
539                 RtlLeaveCriticalSection(&FindDataHandle->Lock);
540                 RtlDeleteCriticalSection(&FindDataHandle->Lock);
541                 break;
542             }
543 
544             default:
545             {
546                 SetLastError(ERROR_INVALID_HANDLE);
547                 _SEH2_YIELD(return FALSE);
548             }
549         }
550 
551         RtlFreeHeap(RtlGetProcessHeap(), 0, FindDataHandle);
552     }
553     _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
554     {
555         BaseSetLastNTError(_SEH2_GetExceptionCode());
556         _SEH2_YIELD(return FALSE);
557     }
558     _SEH2_END;
559 
560     return TRUE;
561 }
562 
563 
564 /*
565  * @unimplemented
566  */
567 HANDLE
568 WINAPI
569 FindFirstFileExA(IN LPCSTR lpFileName,
570                  IN FINDEX_INFO_LEVELS fInfoLevelId,
571                  OUT LPVOID lpFindFileData,
572                  IN FINDEX_SEARCH_OPS fSearchOp,
573                  LPVOID lpSearchFilter,
574                  IN DWORD dwAdditionalFlags)
575 {
576     HANDLE hSearch;
577     NTSTATUS Status;
578     ANSI_STRING Ansi;
579     UNICODE_STRING UTF8;
580     PUNICODE_STRING lpFileNameW;
581     WIN32_FIND_DATAW FindFileDataW;
582     LPWIN32_FIND_DATAA lpFindFileDataA = (LPWIN32_FIND_DATAA)lpFindFileData;
583 
584     if ((fInfoLevelId != FindExInfoStandard && fInfoLevelId != FindExInfoBasic) ||
585         fSearchOp == FindExSearchLimitToDevices ||
586         dwAdditionalFlags & ~FIND_FIRST_EX_CASE_SENSITIVE /* only supported flag for now */)
587     {
588         SetLastError(fSearchOp == FindExSearchLimitToDevices
589                                 ? ERROR_NOT_SUPPORTED
590                                 : ERROR_INVALID_PARAMETER);
591         return INVALID_HANDLE_VALUE;
592     }
593 
594     lpFileNameW = Basep8BitStringToStaticUnicodeString(lpFileName);
595     if (!lpFileNameW) return INVALID_HANDLE_VALUE;
596 
597     hSearch = FindFirstFileExW(lpFileNameW->Buffer,
598                                fInfoLevelId,
599                                &FindFileDataW,
600                                fSearchOp,
601                                lpSearchFilter,
602                                dwAdditionalFlags);
603     if (hSearch == INVALID_HANDLE_VALUE) return INVALID_HANDLE_VALUE;
604 
605     RtlCopyMemory(lpFindFileDataA,
606                   &FindFileDataW,
607                   FIELD_OFFSET(WIN32_FIND_DATAA, cFileName));
608 
609     RtlInitUnicodeString(&UTF8, FindFileDataW.cFileName);
610     Ansi.Buffer = lpFindFileDataA->cFileName;
611     Ansi.Length = 0;
612     Ansi.MaximumLength = sizeof(lpFindFileDataA->cFileName);
613     Status = BasepUnicodeStringTo8BitString(&Ansi, &UTF8, FALSE);
614     if (!NT_SUCCESS(Status))
615     {
616         FindClose(hSearch);
617         BaseSetLastNTError(Status);
618         return INVALID_HANDLE_VALUE;
619     }
620 
621     if (fInfoLevelId != FindExInfoBasic)
622     {
623         RtlInitUnicodeString(&UTF8, FindFileDataW.cAlternateFileName);
624         Ansi.Buffer = lpFindFileDataA->cAlternateFileName;
625         Ansi.Length = 0;
626         Ansi.MaximumLength = sizeof(lpFindFileDataA->cAlternateFileName);
627         Status = BasepUnicodeStringTo8BitString(&Ansi, &UTF8, FALSE);
628         if (!NT_SUCCESS(Status))
629         {
630             FindClose(hSearch);
631             BaseSetLastNTError(Status);
632             return INVALID_HANDLE_VALUE;
633         }
634     }
635     else
636     {
637         lpFindFileDataA->cAlternateFileName[0] = ANSI_NULL;
638     }
639 
640     return hSearch;
641 }
642 
643 
644 /*
645  * @unimplemented
646  */
647 HANDLE
648 WINAPI
649 FindFirstFileExW(IN LPCWSTR lpFileName,
650                  IN FINDEX_INFO_LEVELS fInfoLevelId,
651                  OUT LPVOID lpFindFileData,
652                  IN FINDEX_SEARCH_OPS fSearchOp,
653                  LPVOID lpSearchFilter,
654                  IN DWORD dwAdditionalFlags)
655 {
656     UNREFERENCED_PARAMETER(lpSearchFilter);
657 
658     TRACE("FindFirstFileExW(lpFileName %S)\n", lpFileName);
659 
660     if ((fInfoLevelId != FindExInfoStandard && fInfoLevelId != FindExInfoBasic) ||
661         fSearchOp == FindExSearchLimitToDevices ||
662         dwAdditionalFlags & ~FIND_FIRST_EX_CASE_SENSITIVE /* only supported flag for now */)
663     {
664         SetLastError(fSearchOp == FindExSearchLimitToDevices
665                                 ? ERROR_NOT_SUPPORTED
666                                 : ERROR_INVALID_PARAMETER);
667         return INVALID_HANDLE_VALUE;
668     }
669 
670     if (fSearchOp == FindExSearchNameMatch ||
671         fSearchOp == FindExSearchLimitToDirectories)
672     {
673         LPWIN32_FIND_DATAW Win32FindData = (LPWIN32_FIND_DATAW)lpFindFileData;
674         PFIND_DATA_HANDLE FindDataHandle;
675         PFIND_FILE_DATA FindFileData;
676 
677         UNICODE_STRING NtPath, FilePattern, FileName;
678         PWSTR NtPathBuffer;
679         RTL_RELATIVE_NAME_U RelativePath;
680         ULONG DeviceNameInfo = 0;
681 
682         NTSTATUS Status;
683         OBJECT_ATTRIBUTES ObjectAttributes;
684         IO_STATUS_BLOCK IoStatusBlock;
685         HANDLE hDirectory = NULL;
686 
687         BOOLEAN HadADot = FALSE;
688 
689         /*
690          * May represent many FILE_BOTH_DIR_INFORMATION
691          * or many FILE_FULL_DIR_INFORMATION structures.
692          * NOTE: NtQueryDirectoryFile requires the buffer to be ULONG-aligned
693          */
694         DECLSPEC_ALIGN(4) BYTE DirectoryInfo[FIND_DATA_SIZE];
695         DIR_INFORMATION DirInfo = { .DirInfo = &DirectoryInfo };
696 
697         RtlInitUnicodeString(&FileName, lpFileName);
698         if (FileName.Length != 0 && FileName.Buffer[FileName.Length / sizeof(WCHAR) - 1] == L'.')
699         {
700             HadADot = TRUE;
701         }
702 
703         if (!RtlDosPathNameToNtPathName_U(lpFileName,
704                                           &NtPath,
705                                           (PCWSTR*)&FilePattern.Buffer,
706                                           &RelativePath))
707         {
708             SetLastError(ERROR_PATH_NOT_FOUND);
709             return INVALID_HANDLE_VALUE;
710         }
711 
712         DPRINT("lpFileName = '%S'\n", lpFileName);
713         DPRINT("FilePattern.Buffer = '%S'\n", FilePattern.Buffer);
714         DPRINT("RelativePath.RelativeName = '%wZ'\n", &RelativePath.RelativeName);
715         DPRINT("NtPath.Buffer = '%S'\n", NtPath.Buffer);
716         DPRINT("NtPath - Before = '%wZ'\n", &NtPath);
717 
718         /* Save the buffer pointer for later, we need to free it! */
719         NtPathBuffer = NtPath.Buffer;
720 
721         /*
722          * Contrary to what Windows does, check NOW whether or not
723          * lpFileName is a DOS driver. Therefore we don't have to
724          * write broken code to check that.
725          */
726         if (!FilePattern.Buffer || !*FilePattern.Buffer)
727         {
728             /* No file pattern specified, or DOS device */
729 
730             DeviceNameInfo = RtlIsDosDeviceName_U(lpFileName);
731             if (DeviceNameInfo != 0)
732             {
733                 RtlReleaseRelativeName(&RelativePath);
734                 RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathBuffer);
735 
736                 /* OK, it's really a DOS device */
737                 CopyDeviceFindData(Win32FindData, lpFileName, DeviceNameInfo);
738                 return FIND_DEVICE_HANDLE;
739             }
740         }
741 
742         /* A file pattern was specified, or it was not a DOS device */
743 
744         /* If there is a file pattern then determine its length */
745         if (FilePattern.Buffer != NULL)
746         {
747             FilePattern.Length = NtPath.Length -
748                 (USHORT)((ULONG_PTR)FilePattern.Buffer - (ULONG_PTR)NtPath.Buffer);
749         }
750         else
751         {
752             FilePattern.Length = 0;
753         }
754         FilePattern.MaximumLength = FilePattern.Length;
755 
756         if (RelativePath.RelativeName.Length != 0 &&
757             RelativePath.RelativeName.Buffer != FilePattern.Buffer)
758         {
759             if (FilePattern.Buffer != NULL)
760             {
761                 /* This is a relative path to RelativePath.ContainingDirectory, adjust NtPath! */
762                 NtPath.Length = NtPath.MaximumLength =
763                     (USHORT)((ULONG_PTR)FilePattern.Buffer - (ULONG_PTR)RelativePath.RelativeName.Buffer);
764                 NtPath.Buffer = RelativePath.RelativeName.Buffer;
765             }
766         }
767         else
768         {
769             /* This is an absolute path, NtPath receives the full path */
770             RelativePath.ContainingDirectory = NULL;
771             if (FilePattern.Buffer != NULL)
772             {
773                 NtPath.Length = NtPath.MaximumLength =
774                     (USHORT)((ULONG_PTR)FilePattern.Buffer - (ULONG_PTR)NtPath.Buffer);
775             }
776         }
777 
778         DPRINT("NtPath - After = '%wZ'\n", &NtPath);
779         DPRINT("FilePattern = '%wZ'\n", &FilePattern);
780         DPRINT("RelativeTo = 0x%p\n", RelativePath.ContainingDirectory);
781 
782         InitializeObjectAttributes(&ObjectAttributes,
783                                    &NtPath,
784                                    (dwAdditionalFlags & FIND_FIRST_EX_CASE_SENSITIVE) ? 0 : OBJ_CASE_INSENSITIVE,
785                                    RelativePath.ContainingDirectory,
786                                    NULL);
787 
788         Status = NtOpenFile(&hDirectory,
789                             FILE_LIST_DIRECTORY | SYNCHRONIZE,
790                             &ObjectAttributes,
791                             &IoStatusBlock,
792                             FILE_SHARE_READ | FILE_SHARE_WRITE,
793                             FILE_DIRECTORY_FILE | FILE_SYNCHRONOUS_IO_NONALERT);
794 
795         if (!NT_SUCCESS(Status))
796         {
797             RtlReleaseRelativeName(&RelativePath);
798             RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathBuffer);
799 
800             /* Adjust the last error codes */
801             if (Status == STATUS_OBJECT_NAME_NOT_FOUND)
802                 Status = STATUS_OBJECT_PATH_NOT_FOUND;
803             else if (Status == STATUS_OBJECT_TYPE_MISMATCH)
804                 Status = STATUS_OBJECT_PATH_NOT_FOUND;
805 
806             BaseSetLastNTError(Status);
807             return INVALID_HANDLE_VALUE;
808         }
809 
810         /*
811          * Fail if there is not any file pattern,
812          * since we are not looking for a device.
813          */
814         if (FilePattern.Length == 0)
815         {
816             NtClose(hDirectory);
817             RtlReleaseRelativeName(&RelativePath);
818             RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathBuffer);
819 
820             SetLastError(ERROR_FILE_NOT_FOUND);
821             return INVALID_HANDLE_VALUE;
822         }
823 
824         /* Change pattern: "*.*" --> "*" */
825         if (FilePattern.Length == 6 &&
826             RtlCompareMemory(FilePattern.Buffer, L"*.*", 6) == 6)
827         {
828             FilePattern.Length = 2;
829         }
830         else
831         {
832             /* Translate wildcard from "real" world to DOS world for lower interpretation */
833             USHORT PatternIndex = 0;
834             while (PatternIndex < FilePattern.Length / sizeof(WCHAR))
835             {
836                 if (PatternIndex > 0)
837                 {
838                     if (FilePattern.Buffer[PatternIndex] == L'.' &&
839                         FilePattern.Buffer[PatternIndex - 1] == L'*')
840                     {
841                         FilePattern.Buffer[PatternIndex - 1] = L'<';
842                     }
843                 }
844 
845                 if (FilePattern.Buffer[PatternIndex] == L'?')
846                 {
847                     FilePattern.Buffer[PatternIndex] = L'>';
848                     if (PatternIndex > 0)
849                     {
850                         if (FilePattern.Buffer[PatternIndex - 1] == L'.')
851                         {
852                             FilePattern.Buffer[PatternIndex - 1] = L'\"';
853                         }
854                     }
855                 }
856                 else if (FilePattern.Buffer[PatternIndex] == L'*')
857                 {
858                     if (PatternIndex > 0)
859                     {
860                         if (FilePattern.Buffer[PatternIndex - 1] == L'.')
861                         {
862                             FilePattern.Buffer[PatternIndex - 1] = L'\"';
863                         }
864                     }
865                 }
866 
867                 PatternIndex++;
868             }
869 
870             /* Handle partial wc if our last dot was eaten */
871             if (HadADot)
872             {
873                 if (FilePattern.Buffer[FilePattern.Length / sizeof(WCHAR) - 1] == L'*')
874                 {
875                     FilePattern.Buffer[FilePattern.Length / sizeof(WCHAR) - 1] = L'<';
876                 }
877             }
878         }
879 
880         Status = NtQueryDirectoryFile(hDirectory,
881                                       NULL, NULL, NULL,
882                                       &IoStatusBlock,
883                                       DirInfo.DirInfo, // == &DirectoryInfo
884                                       sizeof(DirectoryInfo),
885                                       (fInfoLevelId == FindExInfoStandard
886                                                      ? FileBothDirectoryInformation
887                                                      : FileFullDirectoryInformation),
888                                       TRUE, /* Return a single entry */
889                                       &FilePattern,
890                                       TRUE);
891 
892         RtlReleaseRelativeName(&RelativePath);
893         RtlFreeHeap(RtlGetProcessHeap(), 0, NtPathBuffer);
894 
895         if (!NT_SUCCESS(Status))
896         {
897             NtClose(hDirectory);
898             BaseSetLastNTError(Status);
899             return INVALID_HANDLE_VALUE;
900         }
901 
902         ASSERT(DirInfo.FullDirInfo->NextEntryOffset == 0);
903 
904         /* Return the information */
905         CopyFindData(Win32FindData, fInfoLevelId, DirInfo);
906 
907         /*
908          * Initialization of the search handle.
909          */
910         FindDataHandle = RtlAllocateHeap(RtlGetProcessHeap(),
911                                          HEAP_ZERO_MEMORY,
912                                          sizeof(FIND_DATA_HANDLE) +
913                                              sizeof(FIND_FILE_DATA));
914         if (!FindDataHandle)
915         {
916             NtClose(hDirectory);
917             SetLastError(ERROR_NOT_ENOUGH_MEMORY);
918             return INVALID_HANDLE_VALUE;
919         }
920 
921         FindDataHandle->Type = FindFile;
922         FindDataHandle->u.FindFileData = (PFIND_FILE_DATA)(FindDataHandle + 1);
923         FindFileData = FindDataHandle->u.FindFileData;
924 
925         FindFileData->Handle = hDirectory;
926         FindFileData->InfoLevel = fInfoLevelId;
927         FindFileData->SearchOp = fSearchOp;
928         FindFileData->HasMoreData = FALSE;
929         FindFileData->NextDirInfo.DirInfo = NULL;
930 
931         /* The critical section must always be initialized */
932         Status = RtlInitializeCriticalSection(&FindDataHandle->Lock);
933         if (!NT_SUCCESS(Status))
934         {
935             NtClose(hDirectory);
936             RtlFreeHeap(RtlGetProcessHeap(), 0, FindDataHandle);
937 
938             BaseSetLastNTError(Status);
939             return INVALID_HANDLE_VALUE;
940         }
941 
942         return (HANDLE)FindDataHandle;
943     }
944     else
945     {
946         SetLastError(ERROR_NOT_SUPPORTED);
947         return INVALID_HANDLE_VALUE;
948     }
949 }
950 
951 
952 /*
953  * @implemented
954  */
955 HANDLE
956 WINAPI
957 FindFirstStreamW(IN LPCWSTR lpFileName,
958                  IN STREAM_INFO_LEVELS InfoLevel,
959                  OUT LPVOID lpFindStreamData,
960                  IN DWORD dwFlags)
961 {
962     PFIND_DATA_HANDLE FindDataHandle = NULL;
963     PFIND_STREAM_DATA FindStreamData;
964     OBJECT_ATTRIBUTES ObjectAttributes;
965     IO_STATUS_BLOCK IoStatusBlock;
966     UNICODE_STRING NtFilePath;
967     HANDLE FileHandle = NULL;
968     NTSTATUS Status;
969     ULONG BufferSize = 0;
970 
971     if (dwFlags != 0 || InfoLevel != FindStreamInfoStandard ||
972         lpFindStreamData == NULL)
973     {
974         SetLastError(ERROR_INVALID_PARAMETER);
975         return INVALID_HANDLE_VALUE;
976     }
977 
978     /* Validate and translate the filename */
979     if (!RtlDosPathNameToNtPathName_U(lpFileName,
980                                       &NtFilePath,
981                                       NULL, NULL))
982     {
983         SetLastError(ERROR_PATH_NOT_FOUND);
984         return INVALID_HANDLE_VALUE;
985     }
986 
987     /* Open the file */
988     InitializeObjectAttributes(&ObjectAttributes,
989                                &NtFilePath,
990                                OBJ_CASE_INSENSITIVE,
991                                NULL,
992                                NULL);
993 
994     Status = NtCreateFile(&FileHandle,
995                           0,
996                           &ObjectAttributes,
997                           &IoStatusBlock,
998                           NULL, 0,
999                           FILE_SHARE_DELETE | FILE_SHARE_READ | FILE_SHARE_WRITE,
1000                           FILE_OPEN,
1001                           0, NULL, 0);
1002     if (!NT_SUCCESS(Status)) goto Cleanup;
1003 
1004     /*
1005      * Initialization of the search handle.
1006      */
1007     FindDataHandle = RtlAllocateHeap(RtlGetProcessHeap(),
1008                                      HEAP_ZERO_MEMORY,
1009                                      sizeof(FIND_DATA_HANDLE) +
1010                                          sizeof(FIND_STREAM_DATA));
1011     if (!FindDataHandle)
1012     {
1013         Status = STATUS_NO_MEMORY;
1014         goto Cleanup;
1015     }
1016 
1017     FindDataHandle->Type = FindStream;
1018     FindDataHandle->u.FindStreamData = (PFIND_STREAM_DATA)(FindDataHandle + 1);
1019     FindStreamData = FindDataHandle->u.FindStreamData;
1020 
1021     FindStreamData->InfoLevel = InfoLevel;
1022     FindStreamData->FileStreamInfo = NULL;
1023     FindStreamData->CurrentInfo = NULL;
1024 
1025     /* The critical section must always be initialized */
1026     Status = RtlInitializeCriticalSection(&FindDataHandle->Lock);
1027     if (!NT_SUCCESS(Status))
1028     {
1029         RtlFreeHeap(RtlGetProcessHeap(), 0, FindDataHandle);
1030         goto Cleanup;
1031     }
1032 
1033     /* Capture all information about the streams */
1034     do
1035     {
1036         BufferSize += 0x1000;
1037 
1038         if (FindStreamData->FileStreamInfo == NULL)
1039         {
1040             FindStreamData->FileStreamInfo = RtlAllocateHeap(RtlGetProcessHeap(),
1041                                                               HEAP_ZERO_MEMORY,
1042                                                               BufferSize);
1043             if (FindStreamData->FileStreamInfo == NULL)
1044             {
1045                 Status = STATUS_NO_MEMORY;
1046                 break;
1047             }
1048         }
1049         else
1050         {
1051             PFILE_STREAM_INFORMATION pfsi;
1052 
1053             pfsi = RtlReAllocateHeap(RtlGetProcessHeap(),
1054                                      0, // HEAP_ZERO_MEMORY,
1055                                      FindStreamData->FileStreamInfo,
1056                                      BufferSize);
1057             if (pfsi == NULL)
1058             {
1059                 Status = STATUS_NO_MEMORY;
1060                 break;
1061             }
1062 
1063             FindStreamData->FileStreamInfo = pfsi;
1064         }
1065 
1066         Status = NtQueryInformationFile(FileHandle,
1067                                         &IoStatusBlock,
1068                                         FindStreamData->FileStreamInfo,
1069                                         BufferSize,
1070                                         FileStreamInformation);
1071 
1072     } while (Status == STATUS_BUFFER_TOO_SMALL);
1073 
1074     if (NT_SUCCESS(Status))
1075     {
1076         /* Select the first stream and return the information */
1077         FindStreamData->CurrentInfo = FindStreamData->FileStreamInfo;
1078         CopyStreamData(FindStreamData, lpFindStreamData);
1079 
1080         /* All done */
1081         Status = STATUS_SUCCESS;
1082     }
1083     else
1084     {
1085         if (FindStreamData->FileStreamInfo)
1086         {
1087             RtlFreeHeap(RtlGetProcessHeap(), 0, FindStreamData->FileStreamInfo);
1088         }
1089 
1090         RtlFreeHeap(RtlGetProcessHeap(), 0, FindDataHandle);
1091     }
1092 
1093 Cleanup:
1094     if (FileHandle) NtClose(FileHandle);
1095 
1096     RtlFreeHeap(RtlGetProcessHeap(), 0, NtFilePath.Buffer);
1097 
1098     if (NT_SUCCESS(Status))
1099     {
1100         return (HANDLE)FindDataHandle;
1101     }
1102     else
1103     {
1104         BaseSetLastNTError(Status);
1105         return INVALID_HANDLE_VALUE;
1106     }
1107 }
1108 
1109 
1110 /*
1111  * @implemented
1112  */
1113 BOOL
1114 WINAPI
1115 FindNextStreamW(IN HANDLE hFindStream,
1116                 OUT LPVOID lpFindStreamData)
1117 {
1118     PFIND_DATA_HANDLE FindDataHandle = (PFIND_DATA_HANDLE)hFindStream;
1119     PFIND_STREAM_DATA FindStreamData;
1120 
1121     if (hFindStream == NULL || hFindStream == INVALID_HANDLE_VALUE ||
1122         FindDataHandle->Type != FindStream)
1123     {
1124         SetLastError(ERROR_INVALID_HANDLE);
1125         return FALSE;
1126     }
1127 
1128     RtlEnterCriticalSection(&FindDataHandle->Lock);
1129 
1130     FindStreamData = FindDataHandle->u.FindStreamData;
1131 
1132     /* Select next stream if possible */
1133     if (FindStreamData->CurrentInfo->NextEntryOffset != 0)
1134     {
1135         FindStreamData->CurrentInfo = (PFILE_STREAM_INFORMATION)((ULONG_PTR)FindStreamData->CurrentInfo +
1136                                                                  FindStreamData->CurrentInfo->NextEntryOffset);
1137 
1138         /* Return the information */
1139         CopyStreamData(FindStreamData, lpFindStreamData);
1140 
1141         RtlLeaveCriticalSection(&FindDataHandle->Lock);
1142         return TRUE;
1143     }
1144     else
1145     {
1146         RtlLeaveCriticalSection(&FindDataHandle->Lock);
1147 
1148         SetLastError(ERROR_HANDLE_EOF);
1149         return FALSE;
1150     }
1151 }
1152 
1153 /* EOF */
1154