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
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 int vlc_open (const char *filename
, int flags
, ...)
72 flags
|= O_NOINHERIT
; /* O_CLOEXEC */
73 /* Defaults to binary mode */
74 if ((flags
& O_TEXT
) == 0)
80 int unixmode
= va_arg(ap
, int);
89 * open() cannot open files with non-“ANSI” characters on Windows.
90 * We use _wopen() instead. Same thing for mkdir() and stat().
92 wchar_t *wpath
= widen_path (filename
);
96 int fd
= _wopen (wpath
, flags
, mode
);
101 int vlc_openat (int dir
, const char *filename
, int flags
, ...)
103 (void) dir
; (void) filename
; (void) flags
;
113 FILE *stream
= tmpfile();
117 fd
= vlc_dup(fileno(stream
));
122 #else /* Not currently used */
128 int vlc_close (int fd
)
133 int vlc_mkdir( const char *dirname
, mode_t mode
)
135 wchar_t *wpath
= widen_path (dirname
);
139 int ret
= _wmkdir (wpath
);
145 char *vlc_getcwd (void)
150 wchar_t *wdir
= _wgetcwd (NULL
, 0);
154 char *dir
= FromWide (wdir
);
160 /* Under Windows, these wrappers return the list of drive letters
161 * when called with an empty argument or just '\'. */
162 DIR *vlc_opendir (const char *dirname
)
164 wchar_t *wpath
= widen_path (dirname
);
168 vlc_DIR
*p_dir
= malloc (sizeof (*p_dir
));
169 if (unlikely(p_dir
== NULL
))
175 #if !VLC_WINSTORE_APP
176 /* Special mode to list drive letters */
177 if (wpath
[0] == L
'\0' || (wcscmp (wpath
, L
"\\") == 0))
181 p_dir
->u
.drives
= GetLogicalDrives ();
183 return (void *)p_dir
;
187 assert (wpath
[0]); // wpath[1] is defined
188 p_dir
->u
.insert_dot_dot
= !wcscmp (wpath
+ 1, L
":\\");
190 _WDIR
*wdir
= _wopendir (wpath
);
199 return (void *)p_dir
;
202 const char *vlc_readdir (DIR *dir
)
204 vlc_DIR
*p_dir
= (vlc_DIR
*)dir
;
208 #if !VLC_WINSTORE_APP
209 /* Drive letters mode */
210 if (p_dir
->wdir
== NULL
)
212 DWORD drives
= p_dir
->u
.drives
;
216 return NULL
; /* end */
220 for (i
= 0; !(drives
& 1); i
++)
222 p_dir
->u
.drives
&= ~(1UL << i
);
225 if (asprintf (&p_dir
->entry
, "%c:\\", 'A' + i
) == -1)
230 if (p_dir
->u
.insert_dot_dot
)
232 /* Adds "..", gruik! */
233 p_dir
->u
.insert_dot_dot
= false;
234 p_dir
->entry
= strdup ("..");
238 struct _wdirent
*ent
= _wreaddir (p_dir
->wdir
);
239 p_dir
->entry
= (ent
!= NULL
) ? FromWide (ent
->d_name
) : NULL
;
244 int vlc_stat (const char *filename
, struct stat
*buf
)
246 wchar_t *wpath
= widen_path (filename
);
250 static_assert (sizeof (*buf
) == sizeof (struct _stati64
),
251 "Mismatched struct stat definition.");
253 int ret
= _wstati64 (wpath
, buf
);
258 int vlc_lstat (const char *filename
, struct stat
*buf
)
260 return vlc_stat (filename
, buf
);
263 int vlc_unlink (const char *filename
)
265 wchar_t *wpath
= widen_path (filename
);
269 int ret
= _wunlink (wpath
);
274 int vlc_rename (const char *oldpath
, const char *newpath
)
278 wchar_t *wold
= widen_path (oldpath
), *wnew
= widen_path (newpath
);
279 if (wold
== NULL
|| wnew
== NULL
)
282 if (_wrename (wold
, wnew
) && (errno
== EACCES
|| errno
== EEXIST
))
283 { /* Windows does not allow atomic file replacement */
286 errno
= EACCES
; /* restore errno */
289 if (_wrename (wold
, wnew
))
299 int vlc_dup (int oldfd
)
301 int fd
= dup (oldfd
);
303 setmode (fd
, O_BINARY
);
307 int vlc_dup2(int oldfd
, int newfd
)
309 int fd
= dup2(oldfd
, newfd
);
311 setmode(fd
, O_BINARY
);
315 int vlc_pipe (int fds
[2])
321 return _pipe (fds
, 32768, O_NOINHERIT
| O_BINARY
);
325 ssize_t
vlc_write(int fd
, const void *buf
, size_t len
)
327 return write(fd
, buf
, len
);
330 ssize_t
vlc_writev(int fd
, const struct iovec
*iov
, int count
)
332 (void) fd
; (void) iov
; (void) count
;
333 vlc_assert_unreachable();
336 #include <vlc_network.h>
338 int vlc_socket (int pf
, int type
, int proto
, bool nonblock
)
340 int fd
= socket (pf
, type
, proto
);
345 ioctlsocket (fd
, FIONBIO
, &(unsigned long){ 1 });
349 int vlc_socketpair(int pf
, int type
, int proto
, int fds
[2], bool nonblock
)
351 (void) pf
; (void) type
; (void) proto
; (void) fds
; (void) nonblock
;
356 int vlc_accept (int lfd
, struct sockaddr
*addr
, socklen_t
*alen
, bool nonblock
)
358 int fd
= accept (lfd
, addr
, alen
);
359 if (fd
!= -1 && nonblock
)
360 ioctlsocket (fd
, FIONBIO
, &(unsigned long){ 1 });
364 ssize_t
vlc_send(int fd
, const void *buf
, size_t len
, int flags
)
366 WSABUF wsabuf
= { .buf
= (char *)buf
, .len
= len
};
369 return WSASend(fd
, &wsabuf
, 1, &sent
, flags
,
370 NULL
, NULL
) ? -1 : (ssize_t
)sent
;
373 ssize_t
vlc_sendto(int fd
, const void *buf
, size_t len
, int flags
,
374 const struct sockaddr
*dst
, socklen_t dstlen
)
376 WSABUF wsabuf
= { .buf
= (char *)buf
, .len
= len
};
379 return WSASendTo(fd
, &wsabuf
, 1, &sent
, flags
, dst
, dstlen
,
380 NULL
, NULL
) ? -1 : (ssize_t
)sent
;
383 ssize_t
vlc_sendmsg(int fd
, const struct msghdr
*msg
, int flags
)
385 return sendmsg(fd
, msg
, flags
);