xref: /aosp_15_r20/external/libwebsockets/win32port/dirent/dirent-win32.h (revision 1c60b9aca93fdbc9b5f19b2d2194c91294b22281)
1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker  * Dirent interface for Microsoft Visual Studio
3*1c60b9acSAndroid Build Coastguard Worker  *
4*1c60b9acSAndroid Build Coastguard Worker  * Copyright (C) 1998-2019 Toni Ronkko
5*1c60b9acSAndroid Build Coastguard Worker  * This file is part of dirent.  Dirent may be freely distributed
6*1c60b9acSAndroid Build Coastguard Worker  * under the MIT license.  For all details and documentation, see
7*1c60b9acSAndroid Build Coastguard Worker  * https://github.com/tronkko/dirent
8*1c60b9acSAndroid Build Coastguard Worker  */
9*1c60b9acSAndroid Build Coastguard Worker #ifndef DIRENT_H
10*1c60b9acSAndroid Build Coastguard Worker #define DIRENT_H
11*1c60b9acSAndroid Build Coastguard Worker 
12*1c60b9acSAndroid Build Coastguard Worker /* Hide warnings about unreferenced local functions */
13*1c60b9acSAndroid Build Coastguard Worker #if defined(__clang__)
14*1c60b9acSAndroid Build Coastguard Worker #   pragma clang diagnostic ignored "-Wunused-function"
15*1c60b9acSAndroid Build Coastguard Worker #elif defined(_MSC_VER)
16*1c60b9acSAndroid Build Coastguard Worker #   pragma warning(disable:4505)
17*1c60b9acSAndroid Build Coastguard Worker #elif defined(__GNUC__)
18*1c60b9acSAndroid Build Coastguard Worker #   pragma GCC diagnostic ignored "-Wunused-function"
19*1c60b9acSAndroid Build Coastguard Worker #endif
20*1c60b9acSAndroid Build Coastguard Worker 
21*1c60b9acSAndroid Build Coastguard Worker /*
22*1c60b9acSAndroid Build Coastguard Worker  * Include windows.h without Windows Sockets 1.1 to prevent conflicts with
23*1c60b9acSAndroid Build Coastguard Worker  * Windows Sockets 2.0.
24*1c60b9acSAndroid Build Coastguard Worker  */
25*1c60b9acSAndroid Build Coastguard Worker #ifndef WIN32_LEAN_AND_MEAN
26*1c60b9acSAndroid Build Coastguard Worker #   define WIN32_LEAN_AND_MEAN
27*1c60b9acSAndroid Build Coastguard Worker #endif
28*1c60b9acSAndroid Build Coastguard Worker #include <windows.h>
29*1c60b9acSAndroid Build Coastguard Worker 
30*1c60b9acSAndroid Build Coastguard Worker #include <stdio.h>
31*1c60b9acSAndroid Build Coastguard Worker #include <stdarg.h>
32*1c60b9acSAndroid Build Coastguard Worker #include <wchar.h>
33*1c60b9acSAndroid Build Coastguard Worker #include <string.h>
34*1c60b9acSAndroid Build Coastguard Worker #include <stdlib.h>
35*1c60b9acSAndroid Build Coastguard Worker #include <malloc.h>
36*1c60b9acSAndroid Build Coastguard Worker #include <sys/types.h>
37*1c60b9acSAndroid Build Coastguard Worker #include <sys/stat.h>
38*1c60b9acSAndroid Build Coastguard Worker #include <errno.h>
39*1c60b9acSAndroid Build Coastguard Worker 
40*1c60b9acSAndroid Build Coastguard Worker /* Indicates that d_type field is available in dirent structure */
41*1c60b9acSAndroid Build Coastguard Worker #define _DIRENT_HAVE_D_TYPE
42*1c60b9acSAndroid Build Coastguard Worker 
43*1c60b9acSAndroid Build Coastguard Worker /* Indicates that d_namlen field is available in dirent structure */
44*1c60b9acSAndroid Build Coastguard Worker #define _DIRENT_HAVE_D_NAMLEN
45*1c60b9acSAndroid Build Coastguard Worker 
46*1c60b9acSAndroid Build Coastguard Worker /* Entries missing from MSVC 6.0 */
47*1c60b9acSAndroid Build Coastguard Worker #if !defined(FILE_ATTRIBUTE_DEVICE)
48*1c60b9acSAndroid Build Coastguard Worker #   define FILE_ATTRIBUTE_DEVICE 0x40
49*1c60b9acSAndroid Build Coastguard Worker #endif
50*1c60b9acSAndroid Build Coastguard Worker 
51*1c60b9acSAndroid Build Coastguard Worker /* File type and permission flags for stat(), general mask */
52*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFMT)
53*1c60b9acSAndroid Build Coastguard Worker #   define S_IFMT _S_IFMT
54*1c60b9acSAndroid Build Coastguard Worker #endif
55*1c60b9acSAndroid Build Coastguard Worker 
56*1c60b9acSAndroid Build Coastguard Worker /* Directory bit */
57*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFDIR)
58*1c60b9acSAndroid Build Coastguard Worker #   define S_IFDIR _S_IFDIR
59*1c60b9acSAndroid Build Coastguard Worker #endif
60*1c60b9acSAndroid Build Coastguard Worker 
61*1c60b9acSAndroid Build Coastguard Worker /* Character device bit */
62*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFCHR)
63*1c60b9acSAndroid Build Coastguard Worker #   define S_IFCHR _S_IFCHR
64*1c60b9acSAndroid Build Coastguard Worker #endif
65*1c60b9acSAndroid Build Coastguard Worker 
66*1c60b9acSAndroid Build Coastguard Worker /* Pipe bit */
67*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFFIFO)
68*1c60b9acSAndroid Build Coastguard Worker #   define S_IFFIFO _S_IFFIFO
69*1c60b9acSAndroid Build Coastguard Worker #endif
70*1c60b9acSAndroid Build Coastguard Worker 
71*1c60b9acSAndroid Build Coastguard Worker /* Regular file bit */
72*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFREG)
73*1c60b9acSAndroid Build Coastguard Worker #   define S_IFREG _S_IFREG
74*1c60b9acSAndroid Build Coastguard Worker #endif
75*1c60b9acSAndroid Build Coastguard Worker 
76*1c60b9acSAndroid Build Coastguard Worker /* Read permission */
77*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IREAD)
78*1c60b9acSAndroid Build Coastguard Worker #   define S_IREAD _S_IREAD
79*1c60b9acSAndroid Build Coastguard Worker #endif
80*1c60b9acSAndroid Build Coastguard Worker 
81*1c60b9acSAndroid Build Coastguard Worker /* Write permission */
82*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IWRITE)
83*1c60b9acSAndroid Build Coastguard Worker #   define S_IWRITE _S_IWRITE
84*1c60b9acSAndroid Build Coastguard Worker #endif
85*1c60b9acSAndroid Build Coastguard Worker 
86*1c60b9acSAndroid Build Coastguard Worker /* Execute permission */
87*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IEXEC)
88*1c60b9acSAndroid Build Coastguard Worker #   define S_IEXEC _S_IEXEC
89*1c60b9acSAndroid Build Coastguard Worker #endif
90*1c60b9acSAndroid Build Coastguard Worker 
91*1c60b9acSAndroid Build Coastguard Worker /* Pipe */
92*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFIFO)
93*1c60b9acSAndroid Build Coastguard Worker #   define S_IFIFO _S_IFIFO
94*1c60b9acSAndroid Build Coastguard Worker #endif
95*1c60b9acSAndroid Build Coastguard Worker 
96*1c60b9acSAndroid Build Coastguard Worker /* Block device */
97*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFBLK)
98*1c60b9acSAndroid Build Coastguard Worker #   define S_IFBLK 0
99*1c60b9acSAndroid Build Coastguard Worker #endif
100*1c60b9acSAndroid Build Coastguard Worker 
101*1c60b9acSAndroid Build Coastguard Worker /* Link */
102*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFLNK)
103*1c60b9acSAndroid Build Coastguard Worker #   define S_IFLNK 0
104*1c60b9acSAndroid Build Coastguard Worker #endif
105*1c60b9acSAndroid Build Coastguard Worker 
106*1c60b9acSAndroid Build Coastguard Worker /* Socket */
107*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IFSOCK)
108*1c60b9acSAndroid Build Coastguard Worker #   define S_IFSOCK 0
109*1c60b9acSAndroid Build Coastguard Worker #endif
110*1c60b9acSAndroid Build Coastguard Worker 
111*1c60b9acSAndroid Build Coastguard Worker /* Read user permission */
112*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IRUSR)
113*1c60b9acSAndroid Build Coastguard Worker #   define S_IRUSR S_IREAD
114*1c60b9acSAndroid Build Coastguard Worker #endif
115*1c60b9acSAndroid Build Coastguard Worker 
116*1c60b9acSAndroid Build Coastguard Worker /* Write user permission */
117*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IWUSR)
118*1c60b9acSAndroid Build Coastguard Worker #   define S_IWUSR S_IWRITE
119*1c60b9acSAndroid Build Coastguard Worker #endif
120*1c60b9acSAndroid Build Coastguard Worker 
121*1c60b9acSAndroid Build Coastguard Worker /* Execute user permission */
122*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IXUSR)
123*1c60b9acSAndroid Build Coastguard Worker #   define S_IXUSR 0
124*1c60b9acSAndroid Build Coastguard Worker #endif
125*1c60b9acSAndroid Build Coastguard Worker 
126*1c60b9acSAndroid Build Coastguard Worker /* Read group permission */
127*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IRGRP)
128*1c60b9acSAndroid Build Coastguard Worker #   define S_IRGRP 0
129*1c60b9acSAndroid Build Coastguard Worker #endif
130*1c60b9acSAndroid Build Coastguard Worker 
131*1c60b9acSAndroid Build Coastguard Worker /* Write group permission */
132*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IWGRP)
133*1c60b9acSAndroid Build Coastguard Worker #   define S_IWGRP 0
134*1c60b9acSAndroid Build Coastguard Worker #endif
135*1c60b9acSAndroid Build Coastguard Worker 
136*1c60b9acSAndroid Build Coastguard Worker /* Execute group permission */
137*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IXGRP)
138*1c60b9acSAndroid Build Coastguard Worker #   define S_IXGRP 0
139*1c60b9acSAndroid Build Coastguard Worker #endif
140*1c60b9acSAndroid Build Coastguard Worker 
141*1c60b9acSAndroid Build Coastguard Worker /* Read others permission */
142*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IROTH)
143*1c60b9acSAndroid Build Coastguard Worker #   define S_IROTH 0
144*1c60b9acSAndroid Build Coastguard Worker #endif
145*1c60b9acSAndroid Build Coastguard Worker 
146*1c60b9acSAndroid Build Coastguard Worker /* Write others permission */
147*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IWOTH)
148*1c60b9acSAndroid Build Coastguard Worker #   define S_IWOTH 0
149*1c60b9acSAndroid Build Coastguard Worker #endif
150*1c60b9acSAndroid Build Coastguard Worker 
151*1c60b9acSAndroid Build Coastguard Worker /* Execute others permission */
152*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_IXOTH)
153*1c60b9acSAndroid Build Coastguard Worker #   define S_IXOTH 0
154*1c60b9acSAndroid Build Coastguard Worker #endif
155*1c60b9acSAndroid Build Coastguard Worker 
156*1c60b9acSAndroid Build Coastguard Worker /* Maximum length of file name */
157*1c60b9acSAndroid Build Coastguard Worker #if !defined(PATH_MAX)
158*1c60b9acSAndroid Build Coastguard Worker #   define PATH_MAX MAX_PATH
159*1c60b9acSAndroid Build Coastguard Worker #endif
160*1c60b9acSAndroid Build Coastguard Worker #if !defined(FILENAME_MAX)
161*1c60b9acSAndroid Build Coastguard Worker #   define FILENAME_MAX MAX_PATH
162*1c60b9acSAndroid Build Coastguard Worker #endif
163*1c60b9acSAndroid Build Coastguard Worker #if !defined(NAME_MAX)
164*1c60b9acSAndroid Build Coastguard Worker #   define NAME_MAX FILENAME_MAX
165*1c60b9acSAndroid Build Coastguard Worker #endif
166*1c60b9acSAndroid Build Coastguard Worker 
167*1c60b9acSAndroid Build Coastguard Worker /* File type flags for d_type */
168*1c60b9acSAndroid Build Coastguard Worker #define DT_UNKNOWN 0
169*1c60b9acSAndroid Build Coastguard Worker #define DT_REG S_IFREG
170*1c60b9acSAndroid Build Coastguard Worker #define DT_DIR S_IFDIR
171*1c60b9acSAndroid Build Coastguard Worker #define DT_FIFO S_IFIFO
172*1c60b9acSAndroid Build Coastguard Worker #define DT_SOCK S_IFSOCK
173*1c60b9acSAndroid Build Coastguard Worker #define DT_CHR S_IFCHR
174*1c60b9acSAndroid Build Coastguard Worker #define DT_BLK S_IFBLK
175*1c60b9acSAndroid Build Coastguard Worker #define DT_LNK S_IFLNK
176*1c60b9acSAndroid Build Coastguard Worker 
177*1c60b9acSAndroid Build Coastguard Worker /* Macros for converting between st_mode and d_type */
178*1c60b9acSAndroid Build Coastguard Worker #define IFTODT(mode) ((mode) & S_IFMT)
179*1c60b9acSAndroid Build Coastguard Worker #define DTTOIF(type) (type)
180*1c60b9acSAndroid Build Coastguard Worker 
181*1c60b9acSAndroid Build Coastguard Worker /*
182*1c60b9acSAndroid Build Coastguard Worker  * File type macros.  Note that block devices, sockets and links cannot be
183*1c60b9acSAndroid Build Coastguard Worker  * distinguished on Windows and the macros S_ISBLK, S_ISSOCK and S_ISLNK are
184*1c60b9acSAndroid Build Coastguard Worker  * only defined for compatibility.  These macros should always return false
185*1c60b9acSAndroid Build Coastguard Worker  * on Windows.
186*1c60b9acSAndroid Build Coastguard Worker  */
187*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_ISFIFO)
188*1c60b9acSAndroid Build Coastguard Worker #   define S_ISFIFO(mode) (((mode) & S_IFMT) == S_IFIFO)
189*1c60b9acSAndroid Build Coastguard Worker #endif
190*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_ISDIR)
191*1c60b9acSAndroid Build Coastguard Worker #   define S_ISDIR(mode) (((mode) & S_IFMT) == S_IFDIR)
192*1c60b9acSAndroid Build Coastguard Worker #endif
193*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_ISREG)
194*1c60b9acSAndroid Build Coastguard Worker #   define S_ISREG(mode) (((mode) & S_IFMT) == S_IFREG)
195*1c60b9acSAndroid Build Coastguard Worker #endif
196*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_ISLNK)
197*1c60b9acSAndroid Build Coastguard Worker #   define S_ISLNK(mode) (((mode) & S_IFMT) == S_IFLNK)
198*1c60b9acSAndroid Build Coastguard Worker #endif
199*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_ISSOCK)
200*1c60b9acSAndroid Build Coastguard Worker #   define S_ISSOCK(mode) (((mode) & S_IFMT) == S_IFSOCK)
201*1c60b9acSAndroid Build Coastguard Worker #endif
202*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_ISCHR)
203*1c60b9acSAndroid Build Coastguard Worker #   define S_ISCHR(mode) (((mode) & S_IFMT) == S_IFCHR)
204*1c60b9acSAndroid Build Coastguard Worker #endif
205*1c60b9acSAndroid Build Coastguard Worker #if !defined(S_ISBLK)
206*1c60b9acSAndroid Build Coastguard Worker #   define S_ISBLK(mode) (((mode) & S_IFMT) == S_IFBLK)
207*1c60b9acSAndroid Build Coastguard Worker #endif
208*1c60b9acSAndroid Build Coastguard Worker 
209*1c60b9acSAndroid Build Coastguard Worker /* Return the exact length of the file name without zero terminator */
210*1c60b9acSAndroid Build Coastguard Worker #define _D_EXACT_NAMLEN(p) ((p)->d_namlen)
211*1c60b9acSAndroid Build Coastguard Worker 
212*1c60b9acSAndroid Build Coastguard Worker /* Return the maximum size of a file name */
213*1c60b9acSAndroid Build Coastguard Worker #define _D_ALLOC_NAMLEN(p) ((PATH_MAX)+1)
214*1c60b9acSAndroid Build Coastguard Worker 
215*1c60b9acSAndroid Build Coastguard Worker 
216*1c60b9acSAndroid Build Coastguard Worker #ifdef __cplusplus
217*1c60b9acSAndroid Build Coastguard Worker extern "C" {
218*1c60b9acSAndroid Build Coastguard Worker #endif
219*1c60b9acSAndroid Build Coastguard Worker 
220*1c60b9acSAndroid Build Coastguard Worker 
221*1c60b9acSAndroid Build Coastguard Worker /* Wide-character version */
222*1c60b9acSAndroid Build Coastguard Worker struct _wdirent {
223*1c60b9acSAndroid Build Coastguard Worker     /* Always zero */
224*1c60b9acSAndroid Build Coastguard Worker     long d_ino;
225*1c60b9acSAndroid Build Coastguard Worker 
226*1c60b9acSAndroid Build Coastguard Worker     /* File position within stream */
227*1c60b9acSAndroid Build Coastguard Worker     long d_off;
228*1c60b9acSAndroid Build Coastguard Worker 
229*1c60b9acSAndroid Build Coastguard Worker     /* Structure size */
230*1c60b9acSAndroid Build Coastguard Worker     unsigned short d_reclen;
231*1c60b9acSAndroid Build Coastguard Worker 
232*1c60b9acSAndroid Build Coastguard Worker     /* Length of name without \0 */
233*1c60b9acSAndroid Build Coastguard Worker     size_t d_namlen;
234*1c60b9acSAndroid Build Coastguard Worker 
235*1c60b9acSAndroid Build Coastguard Worker     /* File type */
236*1c60b9acSAndroid Build Coastguard Worker     int d_type;
237*1c60b9acSAndroid Build Coastguard Worker 
238*1c60b9acSAndroid Build Coastguard Worker     /* File name */
239*1c60b9acSAndroid Build Coastguard Worker     wchar_t d_name[PATH_MAX+1];
240*1c60b9acSAndroid Build Coastguard Worker };
241*1c60b9acSAndroid Build Coastguard Worker typedef struct _wdirent _wdirent;
242*1c60b9acSAndroid Build Coastguard Worker 
243*1c60b9acSAndroid Build Coastguard Worker struct _WDIR {
244*1c60b9acSAndroid Build Coastguard Worker     /* Current directory entry */
245*1c60b9acSAndroid Build Coastguard Worker     struct _wdirent ent;
246*1c60b9acSAndroid Build Coastguard Worker 
247*1c60b9acSAndroid Build Coastguard Worker     /* Private file data */
248*1c60b9acSAndroid Build Coastguard Worker     WIN32_FIND_DATAW data;
249*1c60b9acSAndroid Build Coastguard Worker 
250*1c60b9acSAndroid Build Coastguard Worker     /* True if data is valid */
251*1c60b9acSAndroid Build Coastguard Worker     int cached;
252*1c60b9acSAndroid Build Coastguard Worker 
253*1c60b9acSAndroid Build Coastguard Worker     /* Win32 search handle */
254*1c60b9acSAndroid Build Coastguard Worker     HANDLE handle;
255*1c60b9acSAndroid Build Coastguard Worker 
256*1c60b9acSAndroid Build Coastguard Worker     /* Initial directory name */
257*1c60b9acSAndroid Build Coastguard Worker     wchar_t *patt;
258*1c60b9acSAndroid Build Coastguard Worker };
259*1c60b9acSAndroid Build Coastguard Worker typedef struct _WDIR _WDIR;
260*1c60b9acSAndroid Build Coastguard Worker 
261*1c60b9acSAndroid Build Coastguard Worker /* Multi-byte character version */
262*1c60b9acSAndroid Build Coastguard Worker struct dirent {
263*1c60b9acSAndroid Build Coastguard Worker     /* Always zero */
264*1c60b9acSAndroid Build Coastguard Worker     long d_ino;
265*1c60b9acSAndroid Build Coastguard Worker 
266*1c60b9acSAndroid Build Coastguard Worker     /* File position within stream */
267*1c60b9acSAndroid Build Coastguard Worker     long d_off;
268*1c60b9acSAndroid Build Coastguard Worker 
269*1c60b9acSAndroid Build Coastguard Worker     /* Structure size */
270*1c60b9acSAndroid Build Coastguard Worker     unsigned short d_reclen;
271*1c60b9acSAndroid Build Coastguard Worker 
272*1c60b9acSAndroid Build Coastguard Worker     /* Length of name without \0 */
273*1c60b9acSAndroid Build Coastguard Worker     size_t d_namlen;
274*1c60b9acSAndroid Build Coastguard Worker 
275*1c60b9acSAndroid Build Coastguard Worker     /* File type */
276*1c60b9acSAndroid Build Coastguard Worker     int d_type;
277*1c60b9acSAndroid Build Coastguard Worker 
278*1c60b9acSAndroid Build Coastguard Worker     /* File name */
279*1c60b9acSAndroid Build Coastguard Worker     char d_name[PATH_MAX+1];
280*1c60b9acSAndroid Build Coastguard Worker };
281*1c60b9acSAndroid Build Coastguard Worker typedef struct dirent dirent;
282*1c60b9acSAndroid Build Coastguard Worker 
283*1c60b9acSAndroid Build Coastguard Worker struct DIR {
284*1c60b9acSAndroid Build Coastguard Worker     struct dirent ent;
285*1c60b9acSAndroid Build Coastguard Worker     struct _WDIR *wdirp;
286*1c60b9acSAndroid Build Coastguard Worker };
287*1c60b9acSAndroid Build Coastguard Worker typedef struct DIR DIR;
288*1c60b9acSAndroid Build Coastguard Worker 
289*1c60b9acSAndroid Build Coastguard Worker 
290*1c60b9acSAndroid Build Coastguard Worker /* Dirent functions */
291*1c60b9acSAndroid Build Coastguard Worker static DIR *opendir (const char *dirname);
292*1c60b9acSAndroid Build Coastguard Worker static _WDIR *_wopendir (const wchar_t *dirname);
293*1c60b9acSAndroid Build Coastguard Worker 
294*1c60b9acSAndroid Build Coastguard Worker static struct dirent *readdir (DIR *dirp);
295*1c60b9acSAndroid Build Coastguard Worker static struct _wdirent *_wreaddir (_WDIR *dirp);
296*1c60b9acSAndroid Build Coastguard Worker 
297*1c60b9acSAndroid Build Coastguard Worker static int readdir_r(
298*1c60b9acSAndroid Build Coastguard Worker     DIR *dirp, struct dirent *entry, struct dirent **result);
299*1c60b9acSAndroid Build Coastguard Worker static int _wreaddir_r(
300*1c60b9acSAndroid Build Coastguard Worker     _WDIR *dirp, struct _wdirent *entry, struct _wdirent **result);
301*1c60b9acSAndroid Build Coastguard Worker 
302*1c60b9acSAndroid Build Coastguard Worker static int closedir (DIR *dirp);
303*1c60b9acSAndroid Build Coastguard Worker static int _wclosedir (_WDIR *dirp);
304*1c60b9acSAndroid Build Coastguard Worker 
305*1c60b9acSAndroid Build Coastguard Worker static void rewinddir (DIR* dirp);
306*1c60b9acSAndroid Build Coastguard Worker static void _wrewinddir (_WDIR* dirp);
307*1c60b9acSAndroid Build Coastguard Worker 
308*1c60b9acSAndroid Build Coastguard Worker static int scandir (const char *dirname, struct dirent ***namelist,
309*1c60b9acSAndroid Build Coastguard Worker     int (*filter)(const struct dirent*),
310*1c60b9acSAndroid Build Coastguard Worker     int (*compare)(const struct dirent**, const struct dirent**));
311*1c60b9acSAndroid Build Coastguard Worker 
312*1c60b9acSAndroid Build Coastguard Worker static int alphasort (const struct dirent **a, const struct dirent **b);
313*1c60b9acSAndroid Build Coastguard Worker 
314*1c60b9acSAndroid Build Coastguard Worker static int versionsort (const struct dirent **a, const struct dirent **b);
315*1c60b9acSAndroid Build Coastguard Worker 
316*1c60b9acSAndroid Build Coastguard Worker 
317*1c60b9acSAndroid Build Coastguard Worker /* For compatibility with Symbian */
318*1c60b9acSAndroid Build Coastguard Worker #define wdirent _wdirent
319*1c60b9acSAndroid Build Coastguard Worker #define WDIR _WDIR
320*1c60b9acSAndroid Build Coastguard Worker #define wopendir _wopendir
321*1c60b9acSAndroid Build Coastguard Worker #define wreaddir _wreaddir
322*1c60b9acSAndroid Build Coastguard Worker #define wclosedir _wclosedir
323*1c60b9acSAndroid Build Coastguard Worker #define wrewinddir _wrewinddir
324*1c60b9acSAndroid Build Coastguard Worker 
325*1c60b9acSAndroid Build Coastguard Worker 
326*1c60b9acSAndroid Build Coastguard Worker /* Internal utility functions */
327*1c60b9acSAndroid Build Coastguard Worker static WIN32_FIND_DATAW *dirent_first (_WDIR *dirp);
328*1c60b9acSAndroid Build Coastguard Worker static WIN32_FIND_DATAW *dirent_next (_WDIR *dirp);
329*1c60b9acSAndroid Build Coastguard Worker 
330*1c60b9acSAndroid Build Coastguard Worker static int dirent_mbstowcs_s(
331*1c60b9acSAndroid Build Coastguard Worker     size_t *pReturnValue,
332*1c60b9acSAndroid Build Coastguard Worker     wchar_t *wcstr,
333*1c60b9acSAndroid Build Coastguard Worker     size_t sizeInWords,
334*1c60b9acSAndroid Build Coastguard Worker     const char *mbstr,
335*1c60b9acSAndroid Build Coastguard Worker     size_t count);
336*1c60b9acSAndroid Build Coastguard Worker 
337*1c60b9acSAndroid Build Coastguard Worker static int dirent_wcstombs_s(
338*1c60b9acSAndroid Build Coastguard Worker     size_t *pReturnValue,
339*1c60b9acSAndroid Build Coastguard Worker     char *mbstr,
340*1c60b9acSAndroid Build Coastguard Worker     size_t sizeInBytes,
341*1c60b9acSAndroid Build Coastguard Worker     const wchar_t *wcstr,
342*1c60b9acSAndroid Build Coastguard Worker     size_t count);
343*1c60b9acSAndroid Build Coastguard Worker 
344*1c60b9acSAndroid Build Coastguard Worker static void dirent_set_errno (int error);
345*1c60b9acSAndroid Build Coastguard Worker 
346*1c60b9acSAndroid Build Coastguard Worker 
347*1c60b9acSAndroid Build Coastguard Worker /*
348*1c60b9acSAndroid Build Coastguard Worker  * Open directory stream DIRNAME for read and return a pointer to the
349*1c60b9acSAndroid Build Coastguard Worker  * internal working area that is used to retrieve individual directory
350*1c60b9acSAndroid Build Coastguard Worker  * entries.
351*1c60b9acSAndroid Build Coastguard Worker  */
352*1c60b9acSAndroid Build Coastguard Worker static _WDIR*
_wopendir(const wchar_t * dirname)353*1c60b9acSAndroid Build Coastguard Worker _wopendir(
354*1c60b9acSAndroid Build Coastguard Worker     const wchar_t *dirname)
355*1c60b9acSAndroid Build Coastguard Worker {
356*1c60b9acSAndroid Build Coastguard Worker     _WDIR *dirp;
357*1c60b9acSAndroid Build Coastguard Worker     DWORD n;
358*1c60b9acSAndroid Build Coastguard Worker     wchar_t *p;
359*1c60b9acSAndroid Build Coastguard Worker 
360*1c60b9acSAndroid Build Coastguard Worker     /* Must have directory name */
361*1c60b9acSAndroid Build Coastguard Worker     if (dirname == NULL  ||  dirname[0] == '\0') {
362*1c60b9acSAndroid Build Coastguard Worker         dirent_set_errno (ENOENT);
363*1c60b9acSAndroid Build Coastguard Worker         return NULL;
364*1c60b9acSAndroid Build Coastguard Worker     }
365*1c60b9acSAndroid Build Coastguard Worker 
366*1c60b9acSAndroid Build Coastguard Worker     /* Allocate new _WDIR structure */
367*1c60b9acSAndroid Build Coastguard Worker     dirp = (_WDIR*) malloc (sizeof (struct _WDIR));
368*1c60b9acSAndroid Build Coastguard Worker     if (!dirp) {
369*1c60b9acSAndroid Build Coastguard Worker         return NULL;
370*1c60b9acSAndroid Build Coastguard Worker     }
371*1c60b9acSAndroid Build Coastguard Worker 
372*1c60b9acSAndroid Build Coastguard Worker     /* Reset _WDIR structure */
373*1c60b9acSAndroid Build Coastguard Worker     dirp->handle = INVALID_HANDLE_VALUE;
374*1c60b9acSAndroid Build Coastguard Worker     dirp->patt = NULL;
375*1c60b9acSAndroid Build Coastguard Worker     dirp->cached = 0;
376*1c60b9acSAndroid Build Coastguard Worker 
377*1c60b9acSAndroid Build Coastguard Worker     /*
378*1c60b9acSAndroid Build Coastguard Worker      * Compute the length of full path plus zero terminator
379*1c60b9acSAndroid Build Coastguard Worker      *
380*1c60b9acSAndroid Build Coastguard Worker      * Note that on WinRT there's no way to convert relative paths
381*1c60b9acSAndroid Build Coastguard Worker      * into absolute paths, so just assume it is an absolute path.
382*1c60b9acSAndroid Build Coastguard Worker      */
383*1c60b9acSAndroid Build Coastguard Worker #if WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
384*1c60b9acSAndroid Build Coastguard Worker     /* Desktop */
385*1c60b9acSAndroid Build Coastguard Worker     n = GetFullPathNameW (dirname, 0, NULL, NULL);
386*1c60b9acSAndroid Build Coastguard Worker #else
387*1c60b9acSAndroid Build Coastguard Worker     /* WinRT */
388*1c60b9acSAndroid Build Coastguard Worker     n = wcslen (dirname);
389*1c60b9acSAndroid Build Coastguard Worker #endif
390*1c60b9acSAndroid Build Coastguard Worker 
391*1c60b9acSAndroid Build Coastguard Worker     /* Allocate room for absolute directory name and search pattern */
392*1c60b9acSAndroid Build Coastguard Worker     dirp->patt = (wchar_t*) malloc (sizeof (wchar_t) * n + 16);
393*1c60b9acSAndroid Build Coastguard Worker     if (dirp->patt == NULL) {
394*1c60b9acSAndroid Build Coastguard Worker         goto exit_closedir;
395*1c60b9acSAndroid Build Coastguard Worker     }
396*1c60b9acSAndroid Build Coastguard Worker 
397*1c60b9acSAndroid Build Coastguard Worker     /*
398*1c60b9acSAndroid Build Coastguard Worker      * Convert relative directory name to an absolute one.  This
399*1c60b9acSAndroid Build Coastguard Worker      * allows rewinddir() to function correctly even when current
400*1c60b9acSAndroid Build Coastguard Worker      * working directory is changed between opendir() and rewinddir().
401*1c60b9acSAndroid Build Coastguard Worker      *
402*1c60b9acSAndroid Build Coastguard Worker      * Note that on WinRT there's no way to convert relative paths
403*1c60b9acSAndroid Build Coastguard Worker      * into absolute paths, so just assume it is an absolute path.
404*1c60b9acSAndroid Build Coastguard Worker      */
405*1c60b9acSAndroid Build Coastguard Worker #if WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
406*1c60b9acSAndroid Build Coastguard Worker     /* Desktop */
407*1c60b9acSAndroid Build Coastguard Worker     n = GetFullPathNameW (dirname, n, dirp->patt, NULL);
408*1c60b9acSAndroid Build Coastguard Worker     if (n <= 0) {
409*1c60b9acSAndroid Build Coastguard Worker         goto exit_closedir;
410*1c60b9acSAndroid Build Coastguard Worker     }
411*1c60b9acSAndroid Build Coastguard Worker #else
412*1c60b9acSAndroid Build Coastguard Worker     /* WinRT */
413*1c60b9acSAndroid Build Coastguard Worker     wcsncpy_s (dirp->patt, n+1, dirname, n);
414*1c60b9acSAndroid Build Coastguard Worker #endif
415*1c60b9acSAndroid Build Coastguard Worker 
416*1c60b9acSAndroid Build Coastguard Worker     /* Append search pattern \* to the directory name */
417*1c60b9acSAndroid Build Coastguard Worker     p = dirp->patt + n;
418*1c60b9acSAndroid Build Coastguard Worker     switch (p[-1]) {
419*1c60b9acSAndroid Build Coastguard Worker     case '\\':
420*1c60b9acSAndroid Build Coastguard Worker     case '/':
421*1c60b9acSAndroid Build Coastguard Worker     case ':':
422*1c60b9acSAndroid Build Coastguard Worker         /* Directory ends in path separator, e.g. c:\temp\ */
423*1c60b9acSAndroid Build Coastguard Worker         /*NOP*/;
424*1c60b9acSAndroid Build Coastguard Worker         break;
425*1c60b9acSAndroid Build Coastguard Worker 
426*1c60b9acSAndroid Build Coastguard Worker     default:
427*1c60b9acSAndroid Build Coastguard Worker         /* Directory name doesn't end in path separator */
428*1c60b9acSAndroid Build Coastguard Worker         *p++ = '\\';
429*1c60b9acSAndroid Build Coastguard Worker     }
430*1c60b9acSAndroid Build Coastguard Worker     *p++ = '*';
431*1c60b9acSAndroid Build Coastguard Worker     *p = '\0';
432*1c60b9acSAndroid Build Coastguard Worker 
433*1c60b9acSAndroid Build Coastguard Worker     /* Open directory stream and retrieve the first entry */
434*1c60b9acSAndroid Build Coastguard Worker     if (!dirent_first (dirp)) {
435*1c60b9acSAndroid Build Coastguard Worker         goto exit_closedir;
436*1c60b9acSAndroid Build Coastguard Worker     }
437*1c60b9acSAndroid Build Coastguard Worker 
438*1c60b9acSAndroid Build Coastguard Worker     /* Success */
439*1c60b9acSAndroid Build Coastguard Worker     return dirp;
440*1c60b9acSAndroid Build Coastguard Worker 
441*1c60b9acSAndroid Build Coastguard Worker     /* Failure */
442*1c60b9acSAndroid Build Coastguard Worker exit_closedir:
443*1c60b9acSAndroid Build Coastguard Worker     _wclosedir (dirp);
444*1c60b9acSAndroid Build Coastguard Worker     return NULL;
445*1c60b9acSAndroid Build Coastguard Worker }
446*1c60b9acSAndroid Build Coastguard Worker 
447*1c60b9acSAndroid Build Coastguard Worker /*
448*1c60b9acSAndroid Build Coastguard Worker  * Read next directory entry.
449*1c60b9acSAndroid Build Coastguard Worker  *
450*1c60b9acSAndroid Build Coastguard Worker  * Returns pointer to static directory entry which may be overwritten by
451*1c60b9acSAndroid Build Coastguard Worker  * subsequent calls to _wreaddir().
452*1c60b9acSAndroid Build Coastguard Worker  */
453*1c60b9acSAndroid Build Coastguard Worker static struct _wdirent*
_wreaddir(_WDIR * dirp)454*1c60b9acSAndroid Build Coastguard Worker _wreaddir(
455*1c60b9acSAndroid Build Coastguard Worker     _WDIR *dirp)
456*1c60b9acSAndroid Build Coastguard Worker {
457*1c60b9acSAndroid Build Coastguard Worker     struct _wdirent *entry;
458*1c60b9acSAndroid Build Coastguard Worker 
459*1c60b9acSAndroid Build Coastguard Worker     /*
460*1c60b9acSAndroid Build Coastguard Worker      * Read directory entry to buffer.  We can safely ignore the return value
461*1c60b9acSAndroid Build Coastguard Worker      * as entry will be set to NULL in case of error.
462*1c60b9acSAndroid Build Coastguard Worker      */
463*1c60b9acSAndroid Build Coastguard Worker     (void) _wreaddir_r (dirp, &dirp->ent, &entry);
464*1c60b9acSAndroid Build Coastguard Worker 
465*1c60b9acSAndroid Build Coastguard Worker     /* Return pointer to statically allocated directory entry */
466*1c60b9acSAndroid Build Coastguard Worker     return entry;
467*1c60b9acSAndroid Build Coastguard Worker }
468*1c60b9acSAndroid Build Coastguard Worker 
469*1c60b9acSAndroid Build Coastguard Worker /*
470*1c60b9acSAndroid Build Coastguard Worker  * Read next directory entry.
471*1c60b9acSAndroid Build Coastguard Worker  *
472*1c60b9acSAndroid Build Coastguard Worker  * Returns zero on success.  If end of directory stream is reached, then sets
473*1c60b9acSAndroid Build Coastguard Worker  * result to NULL and returns zero.
474*1c60b9acSAndroid Build Coastguard Worker  */
475*1c60b9acSAndroid Build Coastguard Worker static int
_wreaddir_r(_WDIR * dirp,struct _wdirent * entry,struct _wdirent ** result)476*1c60b9acSAndroid Build Coastguard Worker _wreaddir_r(
477*1c60b9acSAndroid Build Coastguard Worker     _WDIR *dirp,
478*1c60b9acSAndroid Build Coastguard Worker     struct _wdirent *entry,
479*1c60b9acSAndroid Build Coastguard Worker     struct _wdirent **result)
480*1c60b9acSAndroid Build Coastguard Worker {
481*1c60b9acSAndroid Build Coastguard Worker     WIN32_FIND_DATAW *datap;
482*1c60b9acSAndroid Build Coastguard Worker 
483*1c60b9acSAndroid Build Coastguard Worker     /* Read next directory entry */
484*1c60b9acSAndroid Build Coastguard Worker     datap = dirent_next (dirp);
485*1c60b9acSAndroid Build Coastguard Worker     if (datap) {
486*1c60b9acSAndroid Build Coastguard Worker         size_t n;
487*1c60b9acSAndroid Build Coastguard Worker         DWORD attr;
488*1c60b9acSAndroid Build Coastguard Worker 
489*1c60b9acSAndroid Build Coastguard Worker         /*
490*1c60b9acSAndroid Build Coastguard Worker          * Copy file name as wide-character string.  If the file name is too
491*1c60b9acSAndroid Build Coastguard Worker          * long to fit in to the destination buffer, then truncate file name
492*1c60b9acSAndroid Build Coastguard Worker          * to PATH_MAX characters and zero-terminate the buffer.
493*1c60b9acSAndroid Build Coastguard Worker          */
494*1c60b9acSAndroid Build Coastguard Worker         n = 0;
495*1c60b9acSAndroid Build Coastguard Worker         while (n < PATH_MAX  &&  datap->cFileName[n] != 0) {
496*1c60b9acSAndroid Build Coastguard Worker             entry->d_name[n] = datap->cFileName[n];
497*1c60b9acSAndroid Build Coastguard Worker             n++;
498*1c60b9acSAndroid Build Coastguard Worker         }
499*1c60b9acSAndroid Build Coastguard Worker         entry->d_name[n] = 0;
500*1c60b9acSAndroid Build Coastguard Worker 
501*1c60b9acSAndroid Build Coastguard Worker         /* Length of file name excluding zero terminator */
502*1c60b9acSAndroid Build Coastguard Worker         entry->d_namlen = n;
503*1c60b9acSAndroid Build Coastguard Worker 
504*1c60b9acSAndroid Build Coastguard Worker         /* File type */
505*1c60b9acSAndroid Build Coastguard Worker         attr = datap->dwFileAttributes;
506*1c60b9acSAndroid Build Coastguard Worker         if ((attr & FILE_ATTRIBUTE_DEVICE) != 0) {
507*1c60b9acSAndroid Build Coastguard Worker             entry->d_type = DT_CHR;
508*1c60b9acSAndroid Build Coastguard Worker         } else if ((attr & FILE_ATTRIBUTE_DIRECTORY) != 0) {
509*1c60b9acSAndroid Build Coastguard Worker             entry->d_type = DT_DIR;
510*1c60b9acSAndroid Build Coastguard Worker         } else {
511*1c60b9acSAndroid Build Coastguard Worker             entry->d_type = DT_REG;
512*1c60b9acSAndroid Build Coastguard Worker         }
513*1c60b9acSAndroid Build Coastguard Worker 
514*1c60b9acSAndroid Build Coastguard Worker         /* Reset dummy fields */
515*1c60b9acSAndroid Build Coastguard Worker         entry->d_ino = 0;
516*1c60b9acSAndroid Build Coastguard Worker         entry->d_off = 0;
517*1c60b9acSAndroid Build Coastguard Worker         entry->d_reclen = sizeof (struct _wdirent);
518*1c60b9acSAndroid Build Coastguard Worker 
519*1c60b9acSAndroid Build Coastguard Worker         /* Set result address */
520*1c60b9acSAndroid Build Coastguard Worker         *result = entry;
521*1c60b9acSAndroid Build Coastguard Worker 
522*1c60b9acSAndroid Build Coastguard Worker     } else {
523*1c60b9acSAndroid Build Coastguard Worker 
524*1c60b9acSAndroid Build Coastguard Worker         /* Return NULL to indicate end of directory */
525*1c60b9acSAndroid Build Coastguard Worker         *result = NULL;
526*1c60b9acSAndroid Build Coastguard Worker 
527*1c60b9acSAndroid Build Coastguard Worker     }
528*1c60b9acSAndroid Build Coastguard Worker 
529*1c60b9acSAndroid Build Coastguard Worker     return /*OK*/0;
530*1c60b9acSAndroid Build Coastguard Worker }
531*1c60b9acSAndroid Build Coastguard Worker 
532*1c60b9acSAndroid Build Coastguard Worker /*
533*1c60b9acSAndroid Build Coastguard Worker  * Close directory stream opened by opendir() function.  This invalidates the
534*1c60b9acSAndroid Build Coastguard Worker  * DIR structure as well as any directory entry read previously by
535*1c60b9acSAndroid Build Coastguard Worker  * _wreaddir().
536*1c60b9acSAndroid Build Coastguard Worker  */
537*1c60b9acSAndroid Build Coastguard Worker static int
_wclosedir(_WDIR * dirp)538*1c60b9acSAndroid Build Coastguard Worker _wclosedir(
539*1c60b9acSAndroid Build Coastguard Worker     _WDIR *dirp)
540*1c60b9acSAndroid Build Coastguard Worker {
541*1c60b9acSAndroid Build Coastguard Worker     int ok;
542*1c60b9acSAndroid Build Coastguard Worker     if (dirp) {
543*1c60b9acSAndroid Build Coastguard Worker 
544*1c60b9acSAndroid Build Coastguard Worker         /* Release search handle */
545*1c60b9acSAndroid Build Coastguard Worker         if (dirp->handle != INVALID_HANDLE_VALUE) {
546*1c60b9acSAndroid Build Coastguard Worker             FindClose (dirp->handle);
547*1c60b9acSAndroid Build Coastguard Worker         }
548*1c60b9acSAndroid Build Coastguard Worker 
549*1c60b9acSAndroid Build Coastguard Worker         /* Release search pattern */
550*1c60b9acSAndroid Build Coastguard Worker         free (dirp->patt);
551*1c60b9acSAndroid Build Coastguard Worker 
552*1c60b9acSAndroid Build Coastguard Worker         /* Release directory structure */
553*1c60b9acSAndroid Build Coastguard Worker         free (dirp);
554*1c60b9acSAndroid Build Coastguard Worker         ok = /*success*/0;
555*1c60b9acSAndroid Build Coastguard Worker 
556*1c60b9acSAndroid Build Coastguard Worker     } else {
557*1c60b9acSAndroid Build Coastguard Worker 
558*1c60b9acSAndroid Build Coastguard Worker         /* Invalid directory stream */
559*1c60b9acSAndroid Build Coastguard Worker         dirent_set_errno (EBADF);
560*1c60b9acSAndroid Build Coastguard Worker         ok = /*failure*/-1;
561*1c60b9acSAndroid Build Coastguard Worker 
562*1c60b9acSAndroid Build Coastguard Worker     }
563*1c60b9acSAndroid Build Coastguard Worker     return ok;
564*1c60b9acSAndroid Build Coastguard Worker }
565*1c60b9acSAndroid Build Coastguard Worker 
566*1c60b9acSAndroid Build Coastguard Worker /*
567*1c60b9acSAndroid Build Coastguard Worker  * Rewind directory stream such that _wreaddir() returns the very first
568*1c60b9acSAndroid Build Coastguard Worker  * file name again.
569*1c60b9acSAndroid Build Coastguard Worker  */
570*1c60b9acSAndroid Build Coastguard Worker static void
_wrewinddir(_WDIR * dirp)571*1c60b9acSAndroid Build Coastguard Worker _wrewinddir(
572*1c60b9acSAndroid Build Coastguard Worker     _WDIR* dirp)
573*1c60b9acSAndroid Build Coastguard Worker {
574*1c60b9acSAndroid Build Coastguard Worker     if (dirp) {
575*1c60b9acSAndroid Build Coastguard Worker         /* Release existing search handle */
576*1c60b9acSAndroid Build Coastguard Worker         if (dirp->handle != INVALID_HANDLE_VALUE) {
577*1c60b9acSAndroid Build Coastguard Worker             FindClose (dirp->handle);
578*1c60b9acSAndroid Build Coastguard Worker         }
579*1c60b9acSAndroid Build Coastguard Worker 
580*1c60b9acSAndroid Build Coastguard Worker         /* Open new search handle */
581*1c60b9acSAndroid Build Coastguard Worker         dirent_first (dirp);
582*1c60b9acSAndroid Build Coastguard Worker     }
583*1c60b9acSAndroid Build Coastguard Worker }
584*1c60b9acSAndroid Build Coastguard Worker 
585*1c60b9acSAndroid Build Coastguard Worker /* Get first directory entry (internal) */
586*1c60b9acSAndroid Build Coastguard Worker static WIN32_FIND_DATAW*
dirent_first(_WDIR * dirp)587*1c60b9acSAndroid Build Coastguard Worker dirent_first(
588*1c60b9acSAndroid Build Coastguard Worker     _WDIR *dirp)
589*1c60b9acSAndroid Build Coastguard Worker {
590*1c60b9acSAndroid Build Coastguard Worker     WIN32_FIND_DATAW *datap;
591*1c60b9acSAndroid Build Coastguard Worker     DWORD error;
592*1c60b9acSAndroid Build Coastguard Worker 
593*1c60b9acSAndroid Build Coastguard Worker     /* Open directory and retrieve the first entry */
594*1c60b9acSAndroid Build Coastguard Worker     dirp->handle = FindFirstFileExW(
595*1c60b9acSAndroid Build Coastguard Worker         dirp->patt, FindExInfoStandard, &dirp->data,
596*1c60b9acSAndroid Build Coastguard Worker         FindExSearchNameMatch, NULL, 0);
597*1c60b9acSAndroid Build Coastguard Worker     if (dirp->handle != INVALID_HANDLE_VALUE) {
598*1c60b9acSAndroid Build Coastguard Worker 
599*1c60b9acSAndroid Build Coastguard Worker         /* a directory entry is now waiting in memory */
600*1c60b9acSAndroid Build Coastguard Worker         datap = &dirp->data;
601*1c60b9acSAndroid Build Coastguard Worker         dirp->cached = 1;
602*1c60b9acSAndroid Build Coastguard Worker 
603*1c60b9acSAndroid Build Coastguard Worker     } else {
604*1c60b9acSAndroid Build Coastguard Worker 
605*1c60b9acSAndroid Build Coastguard Worker         /* Failed to open directory: no directory entry in memory */
606*1c60b9acSAndroid Build Coastguard Worker         dirp->cached = 0;
607*1c60b9acSAndroid Build Coastguard Worker         datap = NULL;
608*1c60b9acSAndroid Build Coastguard Worker 
609*1c60b9acSAndroid Build Coastguard Worker         /* Set error code */
610*1c60b9acSAndroid Build Coastguard Worker         error = GetLastError ();
611*1c60b9acSAndroid Build Coastguard Worker         switch (error) {
612*1c60b9acSAndroid Build Coastguard Worker         case ERROR_ACCESS_DENIED:
613*1c60b9acSAndroid Build Coastguard Worker             /* No read access to directory */
614*1c60b9acSAndroid Build Coastguard Worker             dirent_set_errno (EACCES);
615*1c60b9acSAndroid Build Coastguard Worker             break;
616*1c60b9acSAndroid Build Coastguard Worker 
617*1c60b9acSAndroid Build Coastguard Worker         case ERROR_DIRECTORY:
618*1c60b9acSAndroid Build Coastguard Worker             /* Directory name is invalid */
619*1c60b9acSAndroid Build Coastguard Worker             dirent_set_errno (ENOTDIR);
620*1c60b9acSAndroid Build Coastguard Worker             break;
621*1c60b9acSAndroid Build Coastguard Worker 
622*1c60b9acSAndroid Build Coastguard Worker         case ERROR_PATH_NOT_FOUND:
623*1c60b9acSAndroid Build Coastguard Worker         default:
624*1c60b9acSAndroid Build Coastguard Worker             /* Cannot find the file */
625*1c60b9acSAndroid Build Coastguard Worker             dirent_set_errno (ENOENT);
626*1c60b9acSAndroid Build Coastguard Worker         }
627*1c60b9acSAndroid Build Coastguard Worker 
628*1c60b9acSAndroid Build Coastguard Worker     }
629*1c60b9acSAndroid Build Coastguard Worker     return datap;
630*1c60b9acSAndroid Build Coastguard Worker }
631*1c60b9acSAndroid Build Coastguard Worker 
632*1c60b9acSAndroid Build Coastguard Worker /*
633*1c60b9acSAndroid Build Coastguard Worker  * Get next directory entry (internal).
634*1c60b9acSAndroid Build Coastguard Worker  *
635*1c60b9acSAndroid Build Coastguard Worker  * Returns
636*1c60b9acSAndroid Build Coastguard Worker  */
637*1c60b9acSAndroid Build Coastguard Worker static WIN32_FIND_DATAW*
dirent_next(_WDIR * dirp)638*1c60b9acSAndroid Build Coastguard Worker dirent_next(
639*1c60b9acSAndroid Build Coastguard Worker     _WDIR *dirp)
640*1c60b9acSAndroid Build Coastguard Worker {
641*1c60b9acSAndroid Build Coastguard Worker     WIN32_FIND_DATAW *p;
642*1c60b9acSAndroid Build Coastguard Worker 
643*1c60b9acSAndroid Build Coastguard Worker     /* Get next directory entry */
644*1c60b9acSAndroid Build Coastguard Worker     if (dirp->cached != 0) {
645*1c60b9acSAndroid Build Coastguard Worker 
646*1c60b9acSAndroid Build Coastguard Worker         /* A valid directory entry already in memory */
647*1c60b9acSAndroid Build Coastguard Worker         p = &dirp->data;
648*1c60b9acSAndroid Build Coastguard Worker         dirp->cached = 0;
649*1c60b9acSAndroid Build Coastguard Worker 
650*1c60b9acSAndroid Build Coastguard Worker     } else if (dirp->handle != INVALID_HANDLE_VALUE) {
651*1c60b9acSAndroid Build Coastguard Worker 
652*1c60b9acSAndroid Build Coastguard Worker         /* Get the next directory entry from stream */
653*1c60b9acSAndroid Build Coastguard Worker         if (FindNextFileW (dirp->handle, &dirp->data) != FALSE) {
654*1c60b9acSAndroid Build Coastguard Worker             /* Got a file */
655*1c60b9acSAndroid Build Coastguard Worker             p = &dirp->data;
656*1c60b9acSAndroid Build Coastguard Worker         } else {
657*1c60b9acSAndroid Build Coastguard Worker             /* The very last entry has been processed or an error occurred */
658*1c60b9acSAndroid Build Coastguard Worker             FindClose (dirp->handle);
659*1c60b9acSAndroid Build Coastguard Worker             dirp->handle = INVALID_HANDLE_VALUE;
660*1c60b9acSAndroid Build Coastguard Worker             p = NULL;
661*1c60b9acSAndroid Build Coastguard Worker         }
662*1c60b9acSAndroid Build Coastguard Worker 
663*1c60b9acSAndroid Build Coastguard Worker     } else {
664*1c60b9acSAndroid Build Coastguard Worker 
665*1c60b9acSAndroid Build Coastguard Worker         /* End of directory stream reached */
666*1c60b9acSAndroid Build Coastguard Worker         p = NULL;
667*1c60b9acSAndroid Build Coastguard Worker 
668*1c60b9acSAndroid Build Coastguard Worker     }
669*1c60b9acSAndroid Build Coastguard Worker 
670*1c60b9acSAndroid Build Coastguard Worker     return p;
671*1c60b9acSAndroid Build Coastguard Worker }
672*1c60b9acSAndroid Build Coastguard Worker 
673*1c60b9acSAndroid Build Coastguard Worker /*
674*1c60b9acSAndroid Build Coastguard Worker  * Open directory stream using plain old C-string.
675*1c60b9acSAndroid Build Coastguard Worker  */
676*1c60b9acSAndroid Build Coastguard Worker static DIR*
opendir(const char * dirname)677*1c60b9acSAndroid Build Coastguard Worker opendir(
678*1c60b9acSAndroid Build Coastguard Worker     const char *dirname)
679*1c60b9acSAndroid Build Coastguard Worker {
680*1c60b9acSAndroid Build Coastguard Worker     struct DIR *dirp;
681*1c60b9acSAndroid Build Coastguard Worker 
682*1c60b9acSAndroid Build Coastguard Worker     /* Must have directory name */
683*1c60b9acSAndroid Build Coastguard Worker     if (dirname == NULL  ||  dirname[0] == '\0') {
684*1c60b9acSAndroid Build Coastguard Worker         dirent_set_errno (ENOENT);
685*1c60b9acSAndroid Build Coastguard Worker         return NULL;
686*1c60b9acSAndroid Build Coastguard Worker     }
687*1c60b9acSAndroid Build Coastguard Worker 
688*1c60b9acSAndroid Build Coastguard Worker     /* Allocate memory for DIR structure */
689*1c60b9acSAndroid Build Coastguard Worker     dirp = (DIR*) malloc (sizeof (struct DIR));
690*1c60b9acSAndroid Build Coastguard Worker     if (!dirp) {
691*1c60b9acSAndroid Build Coastguard Worker         return NULL;
692*1c60b9acSAndroid Build Coastguard Worker     }
693*1c60b9acSAndroid Build Coastguard Worker     {
694*1c60b9acSAndroid Build Coastguard Worker         int error;
695*1c60b9acSAndroid Build Coastguard Worker         wchar_t wname[PATH_MAX + 1];
696*1c60b9acSAndroid Build Coastguard Worker         size_t n;
697*1c60b9acSAndroid Build Coastguard Worker 
698*1c60b9acSAndroid Build Coastguard Worker         /* Convert directory name to wide-character string */
699*1c60b9acSAndroid Build Coastguard Worker         error = dirent_mbstowcs_s(
700*1c60b9acSAndroid Build Coastguard Worker             &n, wname, PATH_MAX + 1, dirname, PATH_MAX + 1);
701*1c60b9acSAndroid Build Coastguard Worker         if (error) {
702*1c60b9acSAndroid Build Coastguard Worker             /*
703*1c60b9acSAndroid Build Coastguard Worker              * Cannot convert file name to wide-character string.  This
704*1c60b9acSAndroid Build Coastguard Worker              * occurs if the string contains invalid multi-byte sequences or
705*1c60b9acSAndroid Build Coastguard Worker              * the output buffer is too small to contain the resulting
706*1c60b9acSAndroid Build Coastguard Worker              * string.
707*1c60b9acSAndroid Build Coastguard Worker              */
708*1c60b9acSAndroid Build Coastguard Worker             goto exit_free;
709*1c60b9acSAndroid Build Coastguard Worker         }
710*1c60b9acSAndroid Build Coastguard Worker 
711*1c60b9acSAndroid Build Coastguard Worker 
712*1c60b9acSAndroid Build Coastguard Worker         /* Open directory stream using wide-character name */
713*1c60b9acSAndroid Build Coastguard Worker         dirp->wdirp = _wopendir (wname);
714*1c60b9acSAndroid Build Coastguard Worker         if (!dirp->wdirp) {
715*1c60b9acSAndroid Build Coastguard Worker             goto exit_free;
716*1c60b9acSAndroid Build Coastguard Worker         }
717*1c60b9acSAndroid Build Coastguard Worker 
718*1c60b9acSAndroid Build Coastguard Worker     }
719*1c60b9acSAndroid Build Coastguard Worker 
720*1c60b9acSAndroid Build Coastguard Worker     /* Success */
721*1c60b9acSAndroid Build Coastguard Worker     return dirp;
722*1c60b9acSAndroid Build Coastguard Worker 
723*1c60b9acSAndroid Build Coastguard Worker     /* Failure */
724*1c60b9acSAndroid Build Coastguard Worker exit_free:
725*1c60b9acSAndroid Build Coastguard Worker     free (dirp);
726*1c60b9acSAndroid Build Coastguard Worker     return NULL;
727*1c60b9acSAndroid Build Coastguard Worker }
728*1c60b9acSAndroid Build Coastguard Worker 
729*1c60b9acSAndroid Build Coastguard Worker /*
730*1c60b9acSAndroid Build Coastguard Worker  * Read next directory entry.
731*1c60b9acSAndroid Build Coastguard Worker  */
732*1c60b9acSAndroid Build Coastguard Worker static struct dirent*
readdir(DIR * dirp)733*1c60b9acSAndroid Build Coastguard Worker readdir(
734*1c60b9acSAndroid Build Coastguard Worker     DIR *dirp)
735*1c60b9acSAndroid Build Coastguard Worker {
736*1c60b9acSAndroid Build Coastguard Worker     struct dirent *entry;
737*1c60b9acSAndroid Build Coastguard Worker 
738*1c60b9acSAndroid Build Coastguard Worker     /*
739*1c60b9acSAndroid Build Coastguard Worker      * Read directory entry to buffer.  We can safely ignore the return value
740*1c60b9acSAndroid Build Coastguard Worker      * as entry will be set to NULL in case of error.
741*1c60b9acSAndroid Build Coastguard Worker      */
742*1c60b9acSAndroid Build Coastguard Worker     (void) readdir_r (dirp, &dirp->ent, &entry);
743*1c60b9acSAndroid Build Coastguard Worker 
744*1c60b9acSAndroid Build Coastguard Worker     /* Return pointer to statically allocated directory entry */
745*1c60b9acSAndroid Build Coastguard Worker     return entry;
746*1c60b9acSAndroid Build Coastguard Worker }
747*1c60b9acSAndroid Build Coastguard Worker 
748*1c60b9acSAndroid Build Coastguard Worker /*
749*1c60b9acSAndroid Build Coastguard Worker  * Read next directory entry into called-allocated buffer.
750*1c60b9acSAndroid Build Coastguard Worker  *
751*1c60b9acSAndroid Build Coastguard Worker  * Returns zero on success.  If the end of directory stream is reached, then
752*1c60b9acSAndroid Build Coastguard Worker  * sets result to NULL and returns zero.
753*1c60b9acSAndroid Build Coastguard Worker  */
754*1c60b9acSAndroid Build Coastguard Worker static int
readdir_r(DIR * dirp,struct dirent * entry,struct dirent ** result)755*1c60b9acSAndroid Build Coastguard Worker readdir_r(
756*1c60b9acSAndroid Build Coastguard Worker     DIR *dirp,
757*1c60b9acSAndroid Build Coastguard Worker     struct dirent *entry,
758*1c60b9acSAndroid Build Coastguard Worker     struct dirent **result)
759*1c60b9acSAndroid Build Coastguard Worker {
760*1c60b9acSAndroid Build Coastguard Worker     WIN32_FIND_DATAW *datap;
761*1c60b9acSAndroid Build Coastguard Worker 
762*1c60b9acSAndroid Build Coastguard Worker     /* Read next directory entry */
763*1c60b9acSAndroid Build Coastguard Worker     datap = dirent_next (dirp->wdirp);
764*1c60b9acSAndroid Build Coastguard Worker     if (datap) {
765*1c60b9acSAndroid Build Coastguard Worker         size_t n;
766*1c60b9acSAndroid Build Coastguard Worker         int error;
767*1c60b9acSAndroid Build Coastguard Worker 
768*1c60b9acSAndroid Build Coastguard Worker         /* Attempt to convert file name to multi-byte string */
769*1c60b9acSAndroid Build Coastguard Worker         error = dirent_wcstombs_s(
770*1c60b9acSAndroid Build Coastguard Worker             &n, entry->d_name, PATH_MAX + 1, datap->cFileName, PATH_MAX + 1);
771*1c60b9acSAndroid Build Coastguard Worker 
772*1c60b9acSAndroid Build Coastguard Worker         /*
773*1c60b9acSAndroid Build Coastguard Worker          * If the file name cannot be represented by a multi-byte string,
774*1c60b9acSAndroid Build Coastguard Worker          * then attempt to use old 8+3 file name.  This allows traditional
775*1c60b9acSAndroid Build Coastguard Worker          * Unix-code to access some file names despite of unicode
776*1c60b9acSAndroid Build Coastguard Worker          * characters, although file names may seem unfamiliar to the user.
777*1c60b9acSAndroid Build Coastguard Worker          *
778*1c60b9acSAndroid Build Coastguard Worker          * Be ware that the code below cannot come up with a short file
779*1c60b9acSAndroid Build Coastguard Worker          * name unless the file system provides one.  At least
780*1c60b9acSAndroid Build Coastguard Worker          * VirtualBox shared folders fail to do this.
781*1c60b9acSAndroid Build Coastguard Worker          */
782*1c60b9acSAndroid Build Coastguard Worker         if (error  &&  datap->cAlternateFileName[0] != '\0') {
783*1c60b9acSAndroid Build Coastguard Worker             error = dirent_wcstombs_s(
784*1c60b9acSAndroid Build Coastguard Worker                 &n, entry->d_name, PATH_MAX + 1,
785*1c60b9acSAndroid Build Coastguard Worker                 datap->cAlternateFileName, PATH_MAX + 1);
786*1c60b9acSAndroid Build Coastguard Worker         }
787*1c60b9acSAndroid Build Coastguard Worker 
788*1c60b9acSAndroid Build Coastguard Worker         if (!error) {
789*1c60b9acSAndroid Build Coastguard Worker             DWORD attr;
790*1c60b9acSAndroid Build Coastguard Worker 
791*1c60b9acSAndroid Build Coastguard Worker             /* Length of file name excluding zero terminator */
792*1c60b9acSAndroid Build Coastguard Worker             entry->d_namlen = n - 1;
793*1c60b9acSAndroid Build Coastguard Worker 
794*1c60b9acSAndroid Build Coastguard Worker             /* File attributes */
795*1c60b9acSAndroid Build Coastguard Worker             attr = datap->dwFileAttributes;
796*1c60b9acSAndroid Build Coastguard Worker             if ((attr & FILE_ATTRIBUTE_DEVICE) != 0) {
797*1c60b9acSAndroid Build Coastguard Worker                 entry->d_type = DT_CHR;
798*1c60b9acSAndroid Build Coastguard Worker             } else if ((attr & FILE_ATTRIBUTE_DIRECTORY) != 0) {
799*1c60b9acSAndroid Build Coastguard Worker                 entry->d_type = DT_DIR;
800*1c60b9acSAndroid Build Coastguard Worker             } else {
801*1c60b9acSAndroid Build Coastguard Worker                 entry->d_type = DT_REG;
802*1c60b9acSAndroid Build Coastguard Worker             }
803*1c60b9acSAndroid Build Coastguard Worker 
804*1c60b9acSAndroid Build Coastguard Worker             /* Reset dummy fields */
805*1c60b9acSAndroid Build Coastguard Worker             entry->d_ino = 0;
806*1c60b9acSAndroid Build Coastguard Worker             entry->d_off = 0;
807*1c60b9acSAndroid Build Coastguard Worker             entry->d_reclen = sizeof (struct dirent);
808*1c60b9acSAndroid Build Coastguard Worker 
809*1c60b9acSAndroid Build Coastguard Worker         } else {
810*1c60b9acSAndroid Build Coastguard Worker 
811*1c60b9acSAndroid Build Coastguard Worker             /*
812*1c60b9acSAndroid Build Coastguard Worker              * Cannot convert file name to multi-byte string so construct
813*1c60b9acSAndroid Build Coastguard Worker              * an erroneous directory entry and return that.  Note that
814*1c60b9acSAndroid Build Coastguard Worker              * we cannot return NULL as that would stop the processing
815*1c60b9acSAndroid Build Coastguard Worker              * of directory entries completely.
816*1c60b9acSAndroid Build Coastguard Worker              */
817*1c60b9acSAndroid Build Coastguard Worker             entry->d_name[0] = '?';
818*1c60b9acSAndroid Build Coastguard Worker             entry->d_name[1] = '\0';
819*1c60b9acSAndroid Build Coastguard Worker             entry->d_namlen = 1;
820*1c60b9acSAndroid Build Coastguard Worker             entry->d_type = DT_UNKNOWN;
821*1c60b9acSAndroid Build Coastguard Worker             entry->d_ino = 0;
822*1c60b9acSAndroid Build Coastguard Worker             entry->d_off = -1;
823*1c60b9acSAndroid Build Coastguard Worker             entry->d_reclen = 0;
824*1c60b9acSAndroid Build Coastguard Worker 
825*1c60b9acSAndroid Build Coastguard Worker         }
826*1c60b9acSAndroid Build Coastguard Worker 
827*1c60b9acSAndroid Build Coastguard Worker         /* Return pointer to directory entry */
828*1c60b9acSAndroid Build Coastguard Worker         *result = entry;
829*1c60b9acSAndroid Build Coastguard Worker 
830*1c60b9acSAndroid Build Coastguard Worker     } else {
831*1c60b9acSAndroid Build Coastguard Worker 
832*1c60b9acSAndroid Build Coastguard Worker         /* No more directory entries */
833*1c60b9acSAndroid Build Coastguard Worker         *result = NULL;
834*1c60b9acSAndroid Build Coastguard Worker 
835*1c60b9acSAndroid Build Coastguard Worker     }
836*1c60b9acSAndroid Build Coastguard Worker 
837*1c60b9acSAndroid Build Coastguard Worker     return /*OK*/0;
838*1c60b9acSAndroid Build Coastguard Worker }
839*1c60b9acSAndroid Build Coastguard Worker 
840*1c60b9acSAndroid Build Coastguard Worker /*
841*1c60b9acSAndroid Build Coastguard Worker  * Close directory stream.
842*1c60b9acSAndroid Build Coastguard Worker  */
843*1c60b9acSAndroid Build Coastguard Worker static int
closedir(DIR * dirp)844*1c60b9acSAndroid Build Coastguard Worker closedir(
845*1c60b9acSAndroid Build Coastguard Worker     DIR *dirp)
846*1c60b9acSAndroid Build Coastguard Worker {
847*1c60b9acSAndroid Build Coastguard Worker     int ok;
848*1c60b9acSAndroid Build Coastguard Worker     if (dirp) {
849*1c60b9acSAndroid Build Coastguard Worker 
850*1c60b9acSAndroid Build Coastguard Worker         /* Close wide-character directory stream */
851*1c60b9acSAndroid Build Coastguard Worker         ok = _wclosedir (dirp->wdirp);
852*1c60b9acSAndroid Build Coastguard Worker         dirp->wdirp = NULL;
853*1c60b9acSAndroid Build Coastguard Worker 
854*1c60b9acSAndroid Build Coastguard Worker         /* Release multi-byte character version */
855*1c60b9acSAndroid Build Coastguard Worker         free (dirp);
856*1c60b9acSAndroid Build Coastguard Worker 
857*1c60b9acSAndroid Build Coastguard Worker     } else {
858*1c60b9acSAndroid Build Coastguard Worker 
859*1c60b9acSAndroid Build Coastguard Worker         /* Invalid directory stream */
860*1c60b9acSAndroid Build Coastguard Worker         dirent_set_errno (EBADF);
861*1c60b9acSAndroid Build Coastguard Worker         ok = /*failure*/-1;
862*1c60b9acSAndroid Build Coastguard Worker 
863*1c60b9acSAndroid Build Coastguard Worker     }
864*1c60b9acSAndroid Build Coastguard Worker     return ok;
865*1c60b9acSAndroid Build Coastguard Worker }
866*1c60b9acSAndroid Build Coastguard Worker 
867*1c60b9acSAndroid Build Coastguard Worker /*
868*1c60b9acSAndroid Build Coastguard Worker  * Rewind directory stream to beginning.
869*1c60b9acSAndroid Build Coastguard Worker  */
870*1c60b9acSAndroid Build Coastguard Worker static void
rewinddir(DIR * dirp)871*1c60b9acSAndroid Build Coastguard Worker rewinddir(
872*1c60b9acSAndroid Build Coastguard Worker     DIR* dirp)
873*1c60b9acSAndroid Build Coastguard Worker {
874*1c60b9acSAndroid Build Coastguard Worker     /* Rewind wide-character string directory stream */
875*1c60b9acSAndroid Build Coastguard Worker     _wrewinddir (dirp->wdirp);
876*1c60b9acSAndroid Build Coastguard Worker }
877*1c60b9acSAndroid Build Coastguard Worker 
878*1c60b9acSAndroid Build Coastguard Worker /*
879*1c60b9acSAndroid Build Coastguard Worker  * Scan directory for entries.
880*1c60b9acSAndroid Build Coastguard Worker  */
881*1c60b9acSAndroid Build Coastguard Worker static int
scandir(const char * dirname,struct dirent *** namelist,int (* filter)(const struct dirent *),int (* compare)(const struct dirent **,const struct dirent **))882*1c60b9acSAndroid Build Coastguard Worker scandir(
883*1c60b9acSAndroid Build Coastguard Worker     const char *dirname,
884*1c60b9acSAndroid Build Coastguard Worker     struct dirent ***namelist,
885*1c60b9acSAndroid Build Coastguard Worker     int (*filter)(const struct dirent*),
886*1c60b9acSAndroid Build Coastguard Worker     int (*compare)(const struct dirent**, const struct dirent**))
887*1c60b9acSAndroid Build Coastguard Worker {
888*1c60b9acSAndroid Build Coastguard Worker     struct dirent **files = NULL;
889*1c60b9acSAndroid Build Coastguard Worker     size_t size = 0;
890*1c60b9acSAndroid Build Coastguard Worker     size_t allocated = 0;
891*1c60b9acSAndroid Build Coastguard Worker     const size_t init_size = 1;
892*1c60b9acSAndroid Build Coastguard Worker     DIR *dir = NULL;
893*1c60b9acSAndroid Build Coastguard Worker     struct dirent *entry;
894*1c60b9acSAndroid Build Coastguard Worker     struct dirent *tmp = NULL;
895*1c60b9acSAndroid Build Coastguard Worker     size_t i;
896*1c60b9acSAndroid Build Coastguard Worker     int result = 0;
897*1c60b9acSAndroid Build Coastguard Worker 
898*1c60b9acSAndroid Build Coastguard Worker     /* Open directory stream */
899*1c60b9acSAndroid Build Coastguard Worker     dir = opendir (dirname);
900*1c60b9acSAndroid Build Coastguard Worker     if (dir) {
901*1c60b9acSAndroid Build Coastguard Worker 
902*1c60b9acSAndroid Build Coastguard Worker         /* Read directory entries to memory */
903*1c60b9acSAndroid Build Coastguard Worker         while (1) {
904*1c60b9acSAndroid Build Coastguard Worker 
905*1c60b9acSAndroid Build Coastguard Worker             /* Enlarge pointer table to make room for another pointer */
906*1c60b9acSAndroid Build Coastguard Worker             if (size >= allocated) {
907*1c60b9acSAndroid Build Coastguard Worker                 void *p;
908*1c60b9acSAndroid Build Coastguard Worker                 size_t num_entries;
909*1c60b9acSAndroid Build Coastguard Worker 
910*1c60b9acSAndroid Build Coastguard Worker                 /* Compute number of entries in the enlarged pointer table */
911*1c60b9acSAndroid Build Coastguard Worker                 if (size < init_size) {
912*1c60b9acSAndroid Build Coastguard Worker                     /* Allocate initial pointer table */
913*1c60b9acSAndroid Build Coastguard Worker                     num_entries = init_size;
914*1c60b9acSAndroid Build Coastguard Worker                 } else {
915*1c60b9acSAndroid Build Coastguard Worker                     /* Double the size */
916*1c60b9acSAndroid Build Coastguard Worker                     num_entries = size * 2;
917*1c60b9acSAndroid Build Coastguard Worker                 }
918*1c60b9acSAndroid Build Coastguard Worker 
919*1c60b9acSAndroid Build Coastguard Worker                 /* Allocate first pointer table or enlarge existing table */
920*1c60b9acSAndroid Build Coastguard Worker                 p = realloc (files, sizeof (void*) * num_entries);
921*1c60b9acSAndroid Build Coastguard Worker                 if (p != NULL) {
922*1c60b9acSAndroid Build Coastguard Worker                     /* Got the memory */
923*1c60b9acSAndroid Build Coastguard Worker                     files = (dirent**) p;
924*1c60b9acSAndroid Build Coastguard Worker                     allocated = num_entries;
925*1c60b9acSAndroid Build Coastguard Worker                 } else {
926*1c60b9acSAndroid Build Coastguard Worker                     /* Out of memory */
927*1c60b9acSAndroid Build Coastguard Worker                     result = -1;
928*1c60b9acSAndroid Build Coastguard Worker                     break;
929*1c60b9acSAndroid Build Coastguard Worker                 }
930*1c60b9acSAndroid Build Coastguard Worker 
931*1c60b9acSAndroid Build Coastguard Worker             }
932*1c60b9acSAndroid Build Coastguard Worker 
933*1c60b9acSAndroid Build Coastguard Worker             /* Allocate room for temporary directory entry */
934*1c60b9acSAndroid Build Coastguard Worker             if (tmp == NULL) {
935*1c60b9acSAndroid Build Coastguard Worker                 tmp = (struct dirent*) malloc (sizeof (struct dirent));
936*1c60b9acSAndroid Build Coastguard Worker                 if (tmp == NULL) {
937*1c60b9acSAndroid Build Coastguard Worker                     /* Cannot allocate temporary directory entry */
938*1c60b9acSAndroid Build Coastguard Worker                     result = -1;
939*1c60b9acSAndroid Build Coastguard Worker                     break;
940*1c60b9acSAndroid Build Coastguard Worker                 }
941*1c60b9acSAndroid Build Coastguard Worker             }
942*1c60b9acSAndroid Build Coastguard Worker 
943*1c60b9acSAndroid Build Coastguard Worker             /* Read directory entry to temporary area */
944*1c60b9acSAndroid Build Coastguard Worker             if (readdir_r (dir, tmp, &entry) == /*OK*/0) {
945*1c60b9acSAndroid Build Coastguard Worker 
946*1c60b9acSAndroid Build Coastguard Worker                 /* Did we get an entry? */
947*1c60b9acSAndroid Build Coastguard Worker                 if (entry != NULL) {
948*1c60b9acSAndroid Build Coastguard Worker                     int pass;
949*1c60b9acSAndroid Build Coastguard Worker 
950*1c60b9acSAndroid Build Coastguard Worker                     /* Determine whether to include the entry in result */
951*1c60b9acSAndroid Build Coastguard Worker                     if (filter) {
952*1c60b9acSAndroid Build Coastguard Worker                         /* Let the filter function decide */
953*1c60b9acSAndroid Build Coastguard Worker                         pass = filter (tmp);
954*1c60b9acSAndroid Build Coastguard Worker                     } else {
955*1c60b9acSAndroid Build Coastguard Worker                         /* No filter function, include everything */
956*1c60b9acSAndroid Build Coastguard Worker                         pass = 1;
957*1c60b9acSAndroid Build Coastguard Worker                     }
958*1c60b9acSAndroid Build Coastguard Worker 
959*1c60b9acSAndroid Build Coastguard Worker                     if (pass) {
960*1c60b9acSAndroid Build Coastguard Worker                         /* Store the temporary entry to pointer table */
961*1c60b9acSAndroid Build Coastguard Worker                         files[size++] = tmp;
962*1c60b9acSAndroid Build Coastguard Worker                         tmp = NULL;
963*1c60b9acSAndroid Build Coastguard Worker 
964*1c60b9acSAndroid Build Coastguard Worker                         /* Keep up with the number of files */
965*1c60b9acSAndroid Build Coastguard Worker                         result++;
966*1c60b9acSAndroid Build Coastguard Worker                     }
967*1c60b9acSAndroid Build Coastguard Worker 
968*1c60b9acSAndroid Build Coastguard Worker                 } else {
969*1c60b9acSAndroid Build Coastguard Worker 
970*1c60b9acSAndroid Build Coastguard Worker                     /*
971*1c60b9acSAndroid Build Coastguard Worker                      * End of directory stream reached => sort entries and
972*1c60b9acSAndroid Build Coastguard Worker                      * exit.
973*1c60b9acSAndroid Build Coastguard Worker                      */
974*1c60b9acSAndroid Build Coastguard Worker                     qsort (files, size, sizeof (void*),
975*1c60b9acSAndroid Build Coastguard Worker                         (int (*) (const void*, const void*)) compare);
976*1c60b9acSAndroid Build Coastguard Worker                     break;
977*1c60b9acSAndroid Build Coastguard Worker 
978*1c60b9acSAndroid Build Coastguard Worker                 }
979*1c60b9acSAndroid Build Coastguard Worker 
980*1c60b9acSAndroid Build Coastguard Worker             } else {
981*1c60b9acSAndroid Build Coastguard Worker                 /* Error reading directory entry */
982*1c60b9acSAndroid Build Coastguard Worker                 result = /*Error*/ -1;
983*1c60b9acSAndroid Build Coastguard Worker                 break;
984*1c60b9acSAndroid Build Coastguard Worker             }
985*1c60b9acSAndroid Build Coastguard Worker 
986*1c60b9acSAndroid Build Coastguard Worker         }
987*1c60b9acSAndroid Build Coastguard Worker 
988*1c60b9acSAndroid Build Coastguard Worker     } else {
989*1c60b9acSAndroid Build Coastguard Worker         /* Cannot open directory */
990*1c60b9acSAndroid Build Coastguard Worker         result = /*Error*/ -1;
991*1c60b9acSAndroid Build Coastguard Worker     }
992*1c60b9acSAndroid Build Coastguard Worker 
993*1c60b9acSAndroid Build Coastguard Worker     /* Release temporary directory entry */
994*1c60b9acSAndroid Build Coastguard Worker     free (tmp);
995*1c60b9acSAndroid Build Coastguard Worker 
996*1c60b9acSAndroid Build Coastguard Worker     /* Release allocated memory on error */
997*1c60b9acSAndroid Build Coastguard Worker     if (result < 0) {
998*1c60b9acSAndroid Build Coastguard Worker         for (i = 0; i < size; i++) {
999*1c60b9acSAndroid Build Coastguard Worker             free (files[i]);
1000*1c60b9acSAndroid Build Coastguard Worker         }
1001*1c60b9acSAndroid Build Coastguard Worker         free (files);
1002*1c60b9acSAndroid Build Coastguard Worker         files = NULL;
1003*1c60b9acSAndroid Build Coastguard Worker     }
1004*1c60b9acSAndroid Build Coastguard Worker 
1005*1c60b9acSAndroid Build Coastguard Worker     /* Close directory stream */
1006*1c60b9acSAndroid Build Coastguard Worker     if (dir) {
1007*1c60b9acSAndroid Build Coastguard Worker         closedir (dir);
1008*1c60b9acSAndroid Build Coastguard Worker     }
1009*1c60b9acSAndroid Build Coastguard Worker 
1010*1c60b9acSAndroid Build Coastguard Worker     /* Pass pointer table to caller */
1011*1c60b9acSAndroid Build Coastguard Worker     if (namelist) {
1012*1c60b9acSAndroid Build Coastguard Worker         *namelist = files;
1013*1c60b9acSAndroid Build Coastguard Worker     }
1014*1c60b9acSAndroid Build Coastguard Worker     return result;
1015*1c60b9acSAndroid Build Coastguard Worker }
1016*1c60b9acSAndroid Build Coastguard Worker 
1017*1c60b9acSAndroid Build Coastguard Worker /* Alphabetical sorting */
1018*1c60b9acSAndroid Build Coastguard Worker static int
alphasort(const struct dirent ** a,const struct dirent ** b)1019*1c60b9acSAndroid Build Coastguard Worker alphasort(
1020*1c60b9acSAndroid Build Coastguard Worker     const struct dirent **a, const struct dirent **b)
1021*1c60b9acSAndroid Build Coastguard Worker {
1022*1c60b9acSAndroid Build Coastguard Worker     return strcoll ((*a)->d_name, (*b)->d_name);
1023*1c60b9acSAndroid Build Coastguard Worker }
1024*1c60b9acSAndroid Build Coastguard Worker 
1025*1c60b9acSAndroid Build Coastguard Worker /* Sort versions */
1026*1c60b9acSAndroid Build Coastguard Worker static int
versionsort(const struct dirent ** a,const struct dirent ** b)1027*1c60b9acSAndroid Build Coastguard Worker versionsort(
1028*1c60b9acSAndroid Build Coastguard Worker     const struct dirent **a, const struct dirent **b)
1029*1c60b9acSAndroid Build Coastguard Worker {
1030*1c60b9acSAndroid Build Coastguard Worker     /* FIXME: implement strverscmp and use that */
1031*1c60b9acSAndroid Build Coastguard Worker     return alphasort (a, b);
1032*1c60b9acSAndroid Build Coastguard Worker }
1033*1c60b9acSAndroid Build Coastguard Worker 
1034*1c60b9acSAndroid Build Coastguard Worker /* Convert multi-byte string to wide character string */
1035*1c60b9acSAndroid Build Coastguard Worker static int
dirent_mbstowcs_s(size_t * pReturnValue,wchar_t * wcstr,size_t sizeInWords,const char * mbstr,size_t count)1036*1c60b9acSAndroid Build Coastguard Worker dirent_mbstowcs_s(
1037*1c60b9acSAndroid Build Coastguard Worker     size_t *pReturnValue,
1038*1c60b9acSAndroid Build Coastguard Worker     wchar_t *wcstr,
1039*1c60b9acSAndroid Build Coastguard Worker     size_t sizeInWords,
1040*1c60b9acSAndroid Build Coastguard Worker     const char *mbstr,
1041*1c60b9acSAndroid Build Coastguard Worker     size_t count)
1042*1c60b9acSAndroid Build Coastguard Worker {
1043*1c60b9acSAndroid Build Coastguard Worker     int error;
1044*1c60b9acSAndroid Build Coastguard Worker 
1045*1c60b9acSAndroid Build Coastguard Worker #if defined(_MSC_VER)  &&  _MSC_VER >= 1400
1046*1c60b9acSAndroid Build Coastguard Worker 
1047*1c60b9acSAndroid Build Coastguard Worker     /* Microsoft Visual Studio 2005 or later */
1048*1c60b9acSAndroid Build Coastguard Worker     error = mbstowcs_s (pReturnValue, wcstr, sizeInWords, mbstr, count);
1049*1c60b9acSAndroid Build Coastguard Worker 
1050*1c60b9acSAndroid Build Coastguard Worker #else
1051*1c60b9acSAndroid Build Coastguard Worker 
1052*1c60b9acSAndroid Build Coastguard Worker     /* Older Visual Studio or non-Microsoft compiler */
1053*1c60b9acSAndroid Build Coastguard Worker     size_t n;
1054*1c60b9acSAndroid Build Coastguard Worker 
1055*1c60b9acSAndroid Build Coastguard Worker     /* Convert to wide-character string (or count characters) */
1056*1c60b9acSAndroid Build Coastguard Worker     n = mbstowcs (wcstr, mbstr, sizeInWords);
1057*1c60b9acSAndroid Build Coastguard Worker     if (!wcstr  ||  n < count) {
1058*1c60b9acSAndroid Build Coastguard Worker 
1059*1c60b9acSAndroid Build Coastguard Worker         /* Zero-terminate output buffer */
1060*1c60b9acSAndroid Build Coastguard Worker         if (wcstr  &&  sizeInWords) {
1061*1c60b9acSAndroid Build Coastguard Worker             if (n >= sizeInWords) {
1062*1c60b9acSAndroid Build Coastguard Worker                 n = sizeInWords - 1;
1063*1c60b9acSAndroid Build Coastguard Worker             }
1064*1c60b9acSAndroid Build Coastguard Worker             wcstr[n] = 0;
1065*1c60b9acSAndroid Build Coastguard Worker         }
1066*1c60b9acSAndroid Build Coastguard Worker 
1067*1c60b9acSAndroid Build Coastguard Worker         /* Length of resulting multi-byte string WITH zero terminator */
1068*1c60b9acSAndroid Build Coastguard Worker         if (pReturnValue) {
1069*1c60b9acSAndroid Build Coastguard Worker             *pReturnValue = n + 1;
1070*1c60b9acSAndroid Build Coastguard Worker         }
1071*1c60b9acSAndroid Build Coastguard Worker 
1072*1c60b9acSAndroid Build Coastguard Worker         /* Success */
1073*1c60b9acSAndroid Build Coastguard Worker         error = 0;
1074*1c60b9acSAndroid Build Coastguard Worker 
1075*1c60b9acSAndroid Build Coastguard Worker     } else {
1076*1c60b9acSAndroid Build Coastguard Worker 
1077*1c60b9acSAndroid Build Coastguard Worker         /* Could not convert string */
1078*1c60b9acSAndroid Build Coastguard Worker         error = 1;
1079*1c60b9acSAndroid Build Coastguard Worker 
1080*1c60b9acSAndroid Build Coastguard Worker     }
1081*1c60b9acSAndroid Build Coastguard Worker 
1082*1c60b9acSAndroid Build Coastguard Worker #endif
1083*1c60b9acSAndroid Build Coastguard Worker     return error;
1084*1c60b9acSAndroid Build Coastguard Worker }
1085*1c60b9acSAndroid Build Coastguard Worker 
1086*1c60b9acSAndroid Build Coastguard Worker /* Convert wide-character string to multi-byte string */
1087*1c60b9acSAndroid Build Coastguard Worker static int
dirent_wcstombs_s(size_t * pReturnValue,char * mbstr,size_t sizeInBytes,const wchar_t * wcstr,size_t count)1088*1c60b9acSAndroid Build Coastguard Worker dirent_wcstombs_s(
1089*1c60b9acSAndroid Build Coastguard Worker     size_t *pReturnValue,
1090*1c60b9acSAndroid Build Coastguard Worker     char *mbstr,
1091*1c60b9acSAndroid Build Coastguard Worker     size_t sizeInBytes, /* max size of mbstr */
1092*1c60b9acSAndroid Build Coastguard Worker     const wchar_t *wcstr,
1093*1c60b9acSAndroid Build Coastguard Worker     size_t count)
1094*1c60b9acSAndroid Build Coastguard Worker {
1095*1c60b9acSAndroid Build Coastguard Worker     int error;
1096*1c60b9acSAndroid Build Coastguard Worker 
1097*1c60b9acSAndroid Build Coastguard Worker #if defined(_MSC_VER)  &&  _MSC_VER >= 1400
1098*1c60b9acSAndroid Build Coastguard Worker 
1099*1c60b9acSAndroid Build Coastguard Worker     /* Microsoft Visual Studio 2005 or later */
1100*1c60b9acSAndroid Build Coastguard Worker     error = wcstombs_s (pReturnValue, mbstr, sizeInBytes, wcstr, count);
1101*1c60b9acSAndroid Build Coastguard Worker 
1102*1c60b9acSAndroid Build Coastguard Worker #else
1103*1c60b9acSAndroid Build Coastguard Worker 
1104*1c60b9acSAndroid Build Coastguard Worker     /* Older Visual Studio or non-Microsoft compiler */
1105*1c60b9acSAndroid Build Coastguard Worker     size_t n;
1106*1c60b9acSAndroid Build Coastguard Worker 
1107*1c60b9acSAndroid Build Coastguard Worker     /* Convert to multi-byte string (or count the number of bytes needed) */
1108*1c60b9acSAndroid Build Coastguard Worker     n = wcstombs (mbstr, wcstr, sizeInBytes);
1109*1c60b9acSAndroid Build Coastguard Worker     if (!mbstr  ||  n < count) {
1110*1c60b9acSAndroid Build Coastguard Worker 
1111*1c60b9acSAndroid Build Coastguard Worker         /* Zero-terminate output buffer */
1112*1c60b9acSAndroid Build Coastguard Worker         if (mbstr  &&  sizeInBytes) {
1113*1c60b9acSAndroid Build Coastguard Worker             if (n >= sizeInBytes) {
1114*1c60b9acSAndroid Build Coastguard Worker                 n = sizeInBytes - 1;
1115*1c60b9acSAndroid Build Coastguard Worker             }
1116*1c60b9acSAndroid Build Coastguard Worker             mbstr[n] = '\0';
1117*1c60b9acSAndroid Build Coastguard Worker         }
1118*1c60b9acSAndroid Build Coastguard Worker 
1119*1c60b9acSAndroid Build Coastguard Worker         /* Length of resulting multi-bytes string WITH zero-terminator */
1120*1c60b9acSAndroid Build Coastguard Worker         if (pReturnValue) {
1121*1c60b9acSAndroid Build Coastguard Worker             *pReturnValue = n + 1;
1122*1c60b9acSAndroid Build Coastguard Worker         }
1123*1c60b9acSAndroid Build Coastguard Worker 
1124*1c60b9acSAndroid Build Coastguard Worker         /* Success */
1125*1c60b9acSAndroid Build Coastguard Worker         error = 0;
1126*1c60b9acSAndroid Build Coastguard Worker 
1127*1c60b9acSAndroid Build Coastguard Worker     } else {
1128*1c60b9acSAndroid Build Coastguard Worker 
1129*1c60b9acSAndroid Build Coastguard Worker         /* Cannot convert string */
1130*1c60b9acSAndroid Build Coastguard Worker         error = 1;
1131*1c60b9acSAndroid Build Coastguard Worker 
1132*1c60b9acSAndroid Build Coastguard Worker     }
1133*1c60b9acSAndroid Build Coastguard Worker 
1134*1c60b9acSAndroid Build Coastguard Worker #endif
1135*1c60b9acSAndroid Build Coastguard Worker     return error;
1136*1c60b9acSAndroid Build Coastguard Worker }
1137*1c60b9acSAndroid Build Coastguard Worker 
1138*1c60b9acSAndroid Build Coastguard Worker /* Set errno variable */
1139*1c60b9acSAndroid Build Coastguard Worker static void
dirent_set_errno(int error)1140*1c60b9acSAndroid Build Coastguard Worker dirent_set_errno(
1141*1c60b9acSAndroid Build Coastguard Worker     int error)
1142*1c60b9acSAndroid Build Coastguard Worker {
1143*1c60b9acSAndroid Build Coastguard Worker #if defined(_MSC_VER)  &&  _MSC_VER >= 1400
1144*1c60b9acSAndroid Build Coastguard Worker 
1145*1c60b9acSAndroid Build Coastguard Worker     /* Microsoft Visual Studio 2005 and later */
1146*1c60b9acSAndroid Build Coastguard Worker     _set_errno (error);
1147*1c60b9acSAndroid Build Coastguard Worker 
1148*1c60b9acSAndroid Build Coastguard Worker #else
1149*1c60b9acSAndroid Build Coastguard Worker 
1150*1c60b9acSAndroid Build Coastguard Worker     /* Non-Microsoft compiler or older Microsoft compiler */
1151*1c60b9acSAndroid Build Coastguard Worker     errno = error;
1152*1c60b9acSAndroid Build Coastguard Worker 
1153*1c60b9acSAndroid Build Coastguard Worker #endif
1154*1c60b9acSAndroid Build Coastguard Worker }
1155*1c60b9acSAndroid Build Coastguard Worker 
1156*1c60b9acSAndroid Build Coastguard Worker 
1157*1c60b9acSAndroid Build Coastguard Worker #ifdef __cplusplus
1158*1c60b9acSAndroid Build Coastguard Worker }
1159*1c60b9acSAndroid Build Coastguard Worker #endif
1160*1c60b9acSAndroid Build Coastguard Worker #endif /*DIRENT_H*/
1161