1 /*****************************************************************************
2 * filesystem.c: Windows file system helpers
3 *****************************************************************************
4 * Copyright (C) 2005-2006 VLC authors and VideoLAN
5 * Copyright © 2005-2008 Rémi Denis-Courmont
7 * Authors: Rémi Denis-Courmont <rem # videolan.org>
9 * This program is free software; you can redistribute it and/or modify it
10 * under the terms of the GNU Lesser General Public License as published by
11 * the Free Software Foundation; either version 2.1 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public License
20 * along with this program; if not, write to the Free Software Foundation,
21 * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22 *****************************************************************************/
24 /*****************************************************************************
26 *****************************************************************************/
35 #include <sys/types.h>
42 #include <vlc_common.h>
43 #include <vlc_charset.h>
45 #include "libvlc.h" /* vlc_mkdir */
49 # include <io.h> /* _pipe */
52 static wchar_t *widen_path (const char *path
)
57 wpath
= ToWide (path
);
67 #define CONVERT_PATH(path, wpath, err) \
68 wchar_t *wpath = wide_path(path); \
69 if (wpath == NULL) return (err)
72 int vlc_open (const char *filename
, int flags
, ...)
77 flags
|= O_NOINHERIT
; /* O_CLOEXEC */
78 /* Defaults to binary mode */
79 if ((flags
& O_TEXT
) == 0)
84 mode
= va_arg (ap
, int);
88 * open() cannot open files with non-“ANSI” characters on Windows.
89 * We use _wopen() instead. Same thing for mkdir() and stat().
91 wchar_t *wpath
= widen_path (filename
);
95 int fd
= _wopen (wpath
, flags
, mode
);
100 int vlc_openat (int dir
, const char *filename
, int flags
, ...)
102 (void) dir
; (void) filename
; (void) flags
;
107 int vlc_mkdir( const char *dirname
, mode_t mode
)
109 wchar_t *wpath
= widen_path (dirname
);
113 int ret
= _wmkdir (wpath
);
119 char *vlc_getcwd (void)
121 wchar_t *wdir
= _wgetcwd (NULL
, 0);
125 char *dir
= FromWide (wdir
);
130 /* Under Windows, these wrappers return the list of drive letters
131 * when called with an empty argument or just '\'. */
132 DIR *vlc_opendir (const char *dirname
)
134 wchar_t *wpath
= widen_path (dirname
);
138 vlc_DIR
*p_dir
= malloc (sizeof (*p_dir
));
139 if (unlikely(p_dir
== NULL
))
145 #if !VLC_WINSTORE_APP
146 /* Special mode to list drive letters */
147 if (wpath
[0] == L
'\0' || (wcscmp (wpath
, L
"\\") == 0))
151 p_dir
->u
.drives
= GetLogicalDrives ();
152 return (void *)p_dir
;
156 assert (wpath
[0]); // wpath[1] is defined
157 p_dir
->u
.insert_dot_dot
= !wcscmp (wpath
+ 1, L
":\\");
159 _WDIR
*wdir
= _wopendir (wpath
);
168 return (void *)p_dir
;
171 char *vlc_readdir (DIR *dir
)
173 vlc_DIR
*p_dir
= (vlc_DIR
*)dir
;
177 #if !VLC_WINSTORE_APP
178 /* Drive letters mode */
179 if (p_dir
->wdir
== NULL
)
181 DWORD drives
= p_dir
->u
.drives
;
183 return NULL
; /* end */
186 for (i
= 0; !(drives
& 1); i
++)
188 p_dir
->u
.drives
&= ~(1UL << i
);
191 if (asprintf (&p_dir
->entry
, "%c:\\", 'A' + i
) == -1)
196 if (p_dir
->u
.insert_dot_dot
)
198 /* Adds "..", gruik! */
199 p_dir
->u
.insert_dot_dot
= false;
200 p_dir
->entry
= strdup ("..");
204 struct _wdirent
*ent
= _wreaddir (p_dir
->wdir
);
205 p_dir
->entry
= (ent
!= NULL
) ? FromWide (ent
->d_name
) : NULL
;
210 int vlc_stat (const char *filename
, struct stat
*buf
)
212 wchar_t *wpath
= widen_path (filename
);
216 static_assert (sizeof (*buf
) == sizeof (struct _stati64
),
217 "Mismatched struct stat definition.");
219 int ret
= _wstati64 (wpath
, buf
);
224 int vlc_lstat (const char *filename
, struct stat
*buf
)
226 return vlc_stat (filename
, buf
);
229 int vlc_unlink (const char *filename
)
231 wchar_t *wpath
= widen_path (filename
);
235 int ret
= _wunlink (wpath
);
240 int vlc_rename (const char *oldpath
, const char *newpath
)
244 wchar_t *wold
= widen_path (oldpath
), *wnew
= widen_path (newpath
);
245 if (wold
== NULL
|| wnew
== NULL
)
248 if (_wrename (wold
, wnew
) && (errno
== EACCES
|| errno
== EEXIST
))
249 { /* Windows does not allow atomic file replacement */
252 errno
= EACCES
; /* restore errno */
255 if (_wrename (wold
, wnew
))
265 int vlc_dup (int oldfd
)
267 int fd
= dup (oldfd
);
269 setmode (fd
, O_BINARY
);
273 int vlc_pipe (int fds
[2])
279 return _pipe (fds
, 32768, O_NOINHERIT
| O_BINARY
);
283 #include <vlc_network.h>
285 int vlc_socket (int pf
, int type
, int proto
, bool nonblock
)
287 int fd
= socket (pf
, type
, proto
);
292 ioctlsocket (fd
, FIONBIO
, &(unsigned long){ 1 });
296 int vlc_accept (int lfd
, struct sockaddr
*addr
, socklen_t
*alen
, bool nonblock
)
298 int fd
= accept (lfd
, addr
, alen
);
299 if (fd
!= -1 && nonblock
)
300 ioctlsocket (fd
, FIONBIO
, &(unsigned long){ 1 });