Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
videolan
vlc
Commits
485d0ebd
Commit
485d0ebd
authored
Nov 03, 2010
by
Rémi Denis-Courmont
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Split portable, POSIX and Win32 file system helpers
parent
0512fd3c
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
626 additions
and
518 deletions
+626
-518
src/Makefile.am
src/Makefile.am
+4
-0
src/misc/filesystem.c
src/misc/filesystem.c
+415
-0
src/text/filesystem.c
src/text/filesystem.c
+2
-518
src/win32/filesystem.c
src/win32/filesystem.c
+205
-0
No files found.
src/Makefile.am
View file @
485d0ebd
...
...
@@ -273,6 +273,7 @@ endif
SOURCES_libvlc_darwin
=
\
config/dirs_macos.c
\
misc/atomic.c
\
misc/filesystem.c
\
misc/pthread.c
\
misc/darwin_specific.c
\
$(NULL)
...
...
@@ -280,6 +281,7 @@ SOURCES_libvlc_darwin = \
SOURCES_libvlc_linux
=
\
config/dirs_xdg.c
\
misc/atomic.c
\
misc/filesystem.c
\
misc/pthread.c
\
misc/linux_specific.c
\
$(NULL)
...
...
@@ -288,6 +290,7 @@ SOURCES_libvlc_win32 = \
win32/dirs.c
\
win32/specific.c
\
win32/atomic.c
\
win32/filesystem.c
\
win32/thread.c
\
win32/winsock.c
\
$(NULL)
...
...
@@ -295,6 +298,7 @@ SOURCES_libvlc_win32 = \
SOURCES_libvlc_other
=
\
config/dirs_xdg.c
\
misc/atomic.c
\
misc/filesystem.c
\
misc/pthread.c
\
misc/not_specific.c
...
...
src/misc/filesystem.c
0 → 100644
View file @
485d0ebd
/*****************************************************************************
* filesystem.c: POSIX file system helpers
*****************************************************************************
* Copyright (C) 2005-2006 the VideoLAN team
* Copyright © 2005-2008 Rémi Denis-Courmont
*
* Authors: Rémi Denis-Courmont <rem # videolan.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <vlc_common.h>
#include <vlc_charset.h>
#include <vlc_fs.h>
#include "libvlc.h"
/* vlc_mkdir */
#include <assert.h>
#include <stdio.h>
#include <limits.h>
/* NAME_MAX */
#include <errno.h>
#include <sys/types.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <dirent.h>
#include <sys/socket.h>
#ifndef HAVE_LSTAT
# define lstat( a, b ) stat(a, b)
#endif
/**
* Opens a system file handle.
*
* @param filename file path to open (with UTF-8 encoding)
* @param flags open() flags, see the C library open() documentation
* @return a file handle on success, -1 on error (see errno).
* @note Contrary to standard open(), this function returns file handles
* with the close-on-exec flag enabled.
*/
int
vlc_open
(
const
char
*
filename
,
int
flags
,
...)
{
unsigned
int
mode
=
0
;
va_list
ap
;
va_start
(
ap
,
flags
);
if
(
flags
&
O_CREAT
)
mode
=
va_arg
(
ap
,
unsigned
int
);
va_end
(
ap
);
#ifdef O_CLOEXEC
flags
|=
O_CLOEXEC
;
#endif
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
local_name
==
NULL
)
{
errno
=
ENOENT
;
return
-
1
;
}
int
fd
=
open
(
local_name
,
flags
,
mode
);
if
(
fd
!=
-
1
)
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
LocaleFree
(
local_name
);
return
fd
;
}
/**
* Opens a system file handle relative to an existing directory handle.
*
* @param dir directory file descriptor
* @param filename file path to open (with UTF-8 encoding)
* @param flags open() flags, see the C library open() documentation
* @return a file handle on success, -1 on error (see errno).
* @note Contrary to standard open(), this function returns file handles
* with the close-on-exec flag enabled.
*/
int
vlc_openat
(
int
dir
,
const
char
*
filename
,
int
flags
,
...)
{
unsigned
int
mode
=
0
;
va_list
ap
;
va_start
(
ap
,
flags
);
if
(
flags
&
O_CREAT
)
mode
=
va_arg
(
ap
,
unsigned
int
);
va_end
(
ap
);
#ifdef O_CLOEXEC
flags
|=
O_CLOEXEC
;
#endif
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
local_name
==
NULL
)
{
errno
=
ENOENT
;
return
-
1
;
}
#ifdef HAVE_OPENAT
int
fd
=
openat
(
dir
,
local_name
,
flags
,
mode
);
if
(
fd
!=
-
1
)
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
#else
int
fd
=
-
1
;
errno
=
ENOSYS
;
#endif
LocaleFree
(
local_name
);
return
fd
;
}
/**
* Creates a directory using UTF-8 paths.
*
* @param dirname a UTF-8 string with the name of the directory that you
* want to create.
* @param mode directory permissions
* @return 0 on success, -1 on error (see errno).
*/
int
vlc_mkdir
(
const
char
*
dirname
,
mode_t
mode
)
{
char
*
locname
=
ToLocale
(
dirname
);
if
(
unlikely
(
locname
==
NULL
))
{
errno
=
ENOENT
;
return
-
1
;
}
int
res
=
mkdir
(
locname
,
mode
);
LocaleFree
(
locname
);
return
res
;
}
/**
* Opens a DIR pointer.
*
* @param dirname UTF-8 representation of the directory name
* @return a pointer to the DIR struct, or NULL in case of error.
* Release with standard closedir().
*/
DIR
*
vlc_opendir
(
const
char
*
dirname
)
{
const
char
*
local_name
=
ToLocale
(
dirname
);
if
(
unlikely
(
local_name
==
NULL
))
{
errno
=
ENOENT
;
return
NULL
;
}
DIR
*
dir
=
opendir
(
local_name
);
LocaleFree
(
local_name
);
return
dir
;
}
/**
* Reads the next file name from an open directory.
*
* @param dir The directory that is being read
*
* @return a UTF-8 string of the directory entry. Use free() to release it.
* If there are no more entries in the directory, NULL is returned.
* If an error occurs, errno is set and NULL is returned.
*/
char
*
vlc_readdir
(
DIR
*
dir
)
{
/* Beware that readdir_r() assumes <buf> is large enough to hold the result
* dirent including the file name. A buffer overflow could occur otherwise.
* In particular, pathconf() and _POSIX_NAME_MAX cannot be used here. */
struct
dirent
*
ent
;
char
*
path
=
NULL
;
long
len
=
fpathconf
(
dirfd
(
dir
),
_PC_NAME_MAX
);
if
(
len
==
-
1
)
{
#ifdef NAME_MAX
len
=
NAME_MAX
;
#else
errno
=
ENOMEM
;
return
NULL
;
// OS is broken. There is no sane way to fix this.
#endif
}
len
+=
offsetof
(
struct
dirent
,
d_name
)
+
1
;
struct
dirent
*
buf
=
malloc
(
len
);
if
(
unlikely
(
buf
==
NULL
))
return
NULL
;
int
val
=
readdir_r
(
dir
,
buf
,
&
ent
);
if
(
val
!=
0
)
errno
=
val
;
else
if
(
ent
!=
NULL
)
#ifndef __APPLE__
path
=
strdup
(
ent
->
d_name
);
#else
path
=
FromCharset
(
"UTF-8-MAC"
,
ent
->
d_name
,
strlen
(
ent
->
d_name
));
#endif
free
(
buf
);
return
path
;
}
static
int
vlc_statEx
(
const
char
*
filename
,
struct
stat
*
buf
,
bool
deref
)
{
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
unlikely
(
local_name
==
NULL
))
{
errno
=
ENOENT
;
return
-
1
;
}
int
res
=
deref
?
stat
(
local_name
,
buf
)
:
lstat
(
local_name
,
buf
);
LocaleFree
(
local_name
);
return
res
;
}
/**
* Finds file/inode information, as stat().
* Consider using fstat() instead, if possible.
*
* @param filename UTF-8 file path
*/
int
vlc_stat
(
const
char
*
filename
,
struct
stat
*
buf
)
{
return
vlc_statEx
(
filename
,
buf
,
true
);
}
/**
* Finds file/inode information, as lstat().
* Consider using fstat() instead, if possible.
*
* @param filename UTF-8 file path
*/
int
vlc_lstat
(
const
char
*
filename
,
struct
stat
*
buf
)
{
return
vlc_statEx
(
filename
,
buf
,
false
);
}
/**
* Removes a file.
*
* @param filename a UTF-8 string with the name of the file you want to delete.
* @return A 0 return value indicates success. A -1 return value indicates an
* error, and an error code is stored in errno
*/
int
vlc_unlink
(
const
char
*
filename
)
{
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
unlikely
(
local_name
==
NULL
))
{
errno
=
ENOENT
;
return
-
1
;
}
int
ret
=
unlink
(
local_name
);
LocaleFree
(
local_name
);
return
ret
;
}
/**
* Moves a file atomically. This only works within a single file system.
*
* @param oldpath path to the file before the move
* @param newpath intended path to the file after the move
* @return A 0 return value indicates success. A -1 return value indicates an
* error, and an error code is stored in errno
*/
int
vlc_rename
(
const
char
*
oldpath
,
const
char
*
newpath
)
{
const
char
*
lo
=
ToLocale
(
oldpath
);
if
(
lo
==
NULL
)
goto
error
;
const
char
*
ln
=
ToLocale
(
newpath
);
if
(
ln
==
NULL
)
{
LocaleFree
(
lo
);
error:
errno
=
ENOENT
;
return
-
1
;
}
int
ret
=
rename
(
lo
,
ln
);
LocaleFree
(
lo
);
LocaleFree
(
ln
);
return
ret
;
}
/**
* Duplicates a file descriptor. The new file descriptor has the close-on-exec
* descriptor flag set.
* @return a new file descriptor or -1
*/
int
vlc_dup
(
int
oldfd
)
{
int
newfd
;
#ifdef HAVE_DUP3
/* Unfortunately, dup3() works like dup2(), not like plain dup(). So we
* need such contortion to find the new file descriptor while preserving
* thread safety of the file descriptor table. */
newfd
=
vlc_open
(
"/dev/null"
,
O_RDONLY
);
if
(
likely
(
newfd
!=
-
1
))
{
if
(
likely
(
dup3
(
oldfd
,
newfd
,
O_CLOEXEC
)
==
newfd
))
return
newfd
;
close
(
newfd
);
}
#endif
newfd
=
dup
(
oldfd
);
if
(
likely
(
newfd
!=
-
1
))
fcntl
(
newfd
,
F_SETFD
,
FD_CLOEXEC
);
return
newfd
;
}
#include <vlc_network.h>
/**
* Creates a socket file descriptor. The new file descriptor has the
* close-on-exec flag set.
* @param pf protocol family
* @param type socket type
* @param proto network protocol
* @param nonblock true to create a non-blocking socket
* @return a new file descriptor or -1
*/
int
vlc_socket
(
int
pf
,
int
type
,
int
proto
,
bool
nonblock
)
{
int
fd
;
#ifdef SOCK_CLOEXEC
type
|=
SOCK_CLOEXEC
;
if
(
nonblock
)
type
|=
SOCK_NONBLOCK
;
fd
=
socket
(
pf
,
type
,
proto
);
if
(
fd
!=
-
1
||
errno
!=
EINVAL
)
return
fd
;
type
&=
~
(
SOCK_CLOEXEC
|
SOCK_NONBLOCK
);
#endif
fd
=
socket
(
pf
,
type
,
proto
);
if
(
fd
==
-
1
)
return
-
1
;
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
if
(
nonblock
)
fcntl
(
fd
,
F_SETFL
,
fcntl
(
fd
,
F_GETFL
,
0
)
|
O_NONBLOCK
);
return
fd
;
}
/**
* Accepts an inbound connection request on a listening socket.
* The new file descriptor has the close-on-exec flag set.
* @param lfd listening socket file descriptor
* @param addr pointer to the peer address or NULL [OUT]
* @param alen pointer to the length of the peer address or NULL [OUT]
* @param nonblock whether to put the new socket in non-blocking mode
* @return a new file descriptor, or -1 on error.
*/
int
vlc_accept
(
int
lfd
,
struct
sockaddr
*
addr
,
socklen_t
*
alen
,
bool
nonblock
)
{
#ifdef HAVE_ACCEPT4
int
flags
=
SOCK_CLOEXEC
;
if
(
nonblock
)
flags
|=
SOCK_NONBLOCK
;
do
{
int
fd
=
accept4
(
lfd
,
addr
,
alen
,
flags
);
if
(
fd
!=
-
1
)
return
fd
;
}
while
(
errno
==
EINTR
);
if
(
errno
!=
ENOSYS
)
return
-
1
;
#endif
do
{
int
fd
=
accept
(
lfd
,
addr
,
alen
);
if
(
fd
!=
-
1
)
{
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
if
(
nonblock
)
fcntl
(
fd
,
F_SETFL
,
fcntl
(
fd
,
F_GETFL
,
0
)
|
O_NONBLOCK
);
return
fd
;
}
}
while
(
errno
==
EINTR
);
return
-
1
;
}
src/text/filesystem.c
View file @
485d0ebd
/*****************************************************************************
* filesystem.c:
F
ile system helpers
* filesystem.c:
Common f
ile system helpers
*****************************************************************************
* Copyright (C) 2005-2006 the VideoLAN team
* Copyright © 2005-2008 Rémi Denis-Courmont
...
...
@@ -29,110 +29,15 @@
#endif
#include <vlc_common.h>
#include <vlc_charset.h>
#include <vlc_fs.h>
#include "libvlc.h"
/* vlc_mkdir */
#include <vlc_rand.h>
#include <assert.h>
#include <stdio.h>
#include <limits.h>
/* NAME_MAX */
#include <errno.h>
#include <sys/types.h>
#include <dirent.h>
#ifdef HAVE_SYS_STAT_H
# include <sys/stat.h>
#endif
#ifdef HAVE_FCNTL_H
# include <fcntl.h>
#endif
#ifdef WIN32
# include <io.h>
# include <winsock2.h>
# ifndef UNDER_CE
# include <direct.h>
# else
# include <tchar.h>
# endif
#else
# include <unistd.h>
# include <sys/socket.h>
#endif
#ifndef HAVE_LSTAT
# define lstat( a, b ) stat(a, b)
#endif
#ifdef WIN32
static
int
convert_path
(
const
char
*
restrict
path
,
wchar_t
*
restrict
wpath
)
{
if
(
!
MultiByteToWideChar
(
CP_UTF8
,
0
,
path
,
-
1
,
wpath
,
MAX_PATH
))
{
errno
=
ENOENT
;
return
-
1
;
}
wpath
[
MAX_PATH
]
=
L'\0'
;
return
0
;
}
# define CONVERT_PATH(path, wpath, err) \
wchar_t wpath[MAX_PATH+1]; \
if (convert_path (path, wpath)) \
return (err)
#endif
/**
* Opens a system file handle.
*
* @param filename file path to open (with UTF-8 encoding)
* @param flags open() flags, see the C library open() documentation
* @return a file handle on success, -1 on error (see errno).
* @note Contrary to standard open(), this function returns file handles
* with the close-on-exec flag enabled.
*/
int
vlc_open
(
const
char
*
filename
,
int
flags
,
...)
{
unsigned
int
mode
=
0
;
va_list
ap
;
va_start
(
ap
,
flags
);
if
(
flags
&
O_CREAT
)
mode
=
va_arg
(
ap
,
unsigned
int
);
va_end
(
ap
);
#ifdef O_CLOEXEC
flags
|=
O_CLOEXEC
;
#endif
#ifdef UNDER_CE
/*_open translates to wchar internally on WinCE*/
return
_open
(
filename
,
flags
,
mode
);
#elif defined (WIN32)
/*
* open() cannot open files with non-“ANSI” characters on Windows.
* We use _wopen() instead. Same thing for mkdir() and stat().
*/
CONVERT_PATH
(
filename
,
wpath
,
-
1
);
return
_wopen
(
wpath
,
flags
,
mode
);
#endif
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
local_name
==
NULL
)
{
errno
=
ENOENT
;
return
-
1
;
}
int
fd
=
open
(
local_name
,
flags
,
mode
);
#ifdef HAVE_FCNTL
if
(
fd
!=
-
1
)
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
#endif
LocaleFree
(
local_name
);
return
fd
;
}
#include <fcntl.h>
/**
* Opens a FILE pointer.
...
...
@@ -193,169 +98,6 @@ FILE *vlc_fopen (const char *filename, const char *mode)
return
stream
;
}
/**
* Opens a system file handle relative to an existing directory handle.
*
* @param dir directory file descriptor
* @param filename file path to open (with UTF-8 encoding)
* @param flags open() flags, see the C library open() documentation
* @return a file handle on success, -1 on error (see errno).
* @note Contrary to standard open(), this function returns file handles
* with the close-on-exec flag enabled.
*/
int
vlc_openat
(
int
dir
,
const
char
*
filename
,
int
flags
,
...)
{
unsigned
int
mode
=
0
;
va_list
ap
;
va_start
(
ap
,
flags
);
if
(
flags
&
O_CREAT
)
mode
=
va_arg
(
ap
,
unsigned
int
);
va_end
(
ap
);
#ifdef O_CLOEXEC
flags
|=
O_CLOEXEC
;
#endif
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
local_name
==
NULL
)
{
errno
=
ENOENT
;
return
-
1
;
}
#ifdef HAVE_OPENAT
int
fd
=
openat
(
dir
,
local_name
,
flags
,
mode
);
# ifdef HAVE_FCNTL
if
(
fd
!=
-
1
)
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
# endif
#else
int
fd
=
-
1
;
errno
=
ENOSYS
;
#endif
LocaleFree
(
local_name
);
return
fd
;
}
/**
* Creates a directory using UTF-8 paths.
*
* @param dirname a UTF-8 string with the name of the directory that you
* want to create.
* @param mode directory permissions
* @return 0 on success, -1 on error (see errno).
*/
int
vlc_mkdir
(
const
char
*
dirname
,
mode_t
mode
)
{
#if defined (UNDER_CE)
(
void
)
mode
;
/* mkdir converts internally to wchar */
return
_mkdir
(
dirname
);
#elif defined (WIN32)
(
void
)
mode
;
CONVERT_PATH
(
dirname
,
wpath
,
-
1
);
return
_wmkdir
(
wpath
);
#else
char
*
locname
=
ToLocale
(
dirname
);
int
res
;
if
(
locname
==
NULL
)
{
errno
=
ENOENT
;
return
-
1
;
}
res
=
mkdir
(
locname
,
mode
);
LocaleFree
(
locname
);
return
res
;
#endif
}
/**
* Opens a DIR pointer.
*
* @param dirname UTF-8 representation of the directory name
* @return a pointer to the DIR struct, or NULL in case of error.
* Release with standard closedir().
*/
DIR
*
vlc_opendir
(
const
char
*
dirname
)
{
#ifdef WIN32
CONVERT_PATH
(
dirname
,
wpath
,
NULL
);
return
(
DIR
*
)
vlc_wopendir
(
wpath
);
#else
const
char
*
local_name
=
ToLocale
(
dirname
);
if
(
local_name
!=
NULL
)
{
DIR
*
dir
=
opendir
(
local_name
);
LocaleFree
(
local_name
);
return
dir
;
}
#endif
errno
=
ENOENT
;
return
NULL
;
}
/**
* Reads the next file name from an open directory.
*
* @param dir The directory that is being read
*
* @return a UTF-8 string of the directory entry. Use free() to release it.
* If there are no more entries in the directory, NULL is returned.
* If an error occurs, errno is set and NULL is returned.
*/
char
*
vlc_readdir
(
DIR
*
dir
)
{
#ifdef WIN32
struct
_wdirent
*
ent
=
vlc_wreaddir
(
dir
);
if
(
ent
==
NULL
)
return
NULL
;
return
FromWide
(
ent
->
d_name
);
#else
/* Beware that readdir_r() assumes <buf> is large enough to hold the result
* dirent including the file name. A buffer overflow could occur otherwise.
* In particular, pathconf() and _POSIX_NAME_MAX cannot be used here. */
struct
dirent
*
ent
;
char
*
path
=
NULL
;
long
len
=
fpathconf
(
dirfd
(
dir
),
_PC_NAME_MAX
);
if
(
len
==
-
1
)
{
#ifdef NAME_MAX
len
=
NAME_MAX
;
#else
errno
=
ENOMEM
;
return
NULL
;
// OS is broken. There is no sane way to fix this.
#endif
}
len
+=
offsetof
(
struct
dirent
,
d_name
)
+
1
;
struct
dirent
*
buf
=
malloc
(
len
);
if
(
unlikely
(
buf
==
NULL
))
return
NULL
;
int
val
=
readdir_r
(
dir
,
buf
,
&
ent
);
if
(
val
!=
0
)
errno
=
val
;
else
if
(
ent
!=
NULL
)
#ifndef __APPLE__
path
=
strdup
(
ent
->
d_name
);
#else
path
=
FromCharset
(
"UTF-8-MAC"
,
ent
->
d_name
,
strlen
(
ent
->
d_name
));
#endif
free
(
buf
);
return
path
;
#endif
}
static
int
dummy_select
(
const
char
*
str
)
{
...
...
@@ -453,137 +195,6 @@ int vlc_scandir( const char *dirname, char ***namelist,
return
val
;
}
static
int
vlc_statEx
(
const
char
*
filename
,
struct
stat
*
buf
,
bool
deref
)
{
#ifdef UNDER_CE
/*_stat translates to wchar internally on WinCE*/
return
_stat
(
filename
,
buf
);
#elif defined (WIN32)
CONVERT_PATH
(
filename
,
wpath
,
-
1
);
return
_wstati64
(
wpath
,
buf
);
#endif
#ifdef HAVE_SYS_STAT_H
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
local_name
!=
NULL
)
{
int
res
=
deref
?
stat
(
local_name
,
buf
)
:
lstat
(
local_name
,
buf
);
LocaleFree
(
local_name
);
return
res
;
}
errno
=
ENOENT
;
#endif
return
-
1
;
}
/**
* Finds file/inode information, as stat().
* Consider using fstat() instead, if possible.
*
* @param filename UTF-8 file path
*/
int
vlc_stat
(
const
char
*
filename
,
struct
stat
*
buf
)
{
return
vlc_statEx
(
filename
,
buf
,
true
);
}
/**
* Finds file/inode information, as lstat().
* Consider using fstat() instead, if possible.
*
* @param filename UTF-8 file path
*/
int
vlc_lstat
(
const
char
*
filename
,
struct
stat
*
buf
)
{
return
vlc_statEx
(
filename
,
buf
,
false
);
}
/**
* Removes a file.
*
* @param filename a UTF-8 string with the name of the file you want to delete.
* @return A 0 return value indicates success. A -1 return value indicates an
* error, and an error code is stored in errno
*/
int
vlc_unlink
(
const
char
*
filename
)
{
#ifdef UNDER_CE
/*_open translates to wchar internally on WinCE*/
return
_unlink
(
filename
);
#elif defined (WIN32)
CONVERT_PATH
(
filename
,
wpath
,
-
1
);
return
_wunlink
(
wpath
);
#endif
const
char
*
local_name
=
ToLocale
(
filename
);
if
(
local_name
==
NULL
)
{
errno
=
ENOENT
;
return
-
1
;
}
int
ret
=
unlink
(
local_name
);
LocaleFree
(
local_name
);
return
ret
;
}
/**
* Moves a file atomically. This only works within a single file system.
*
* @param oldpath path to the file before the move
* @param newpath intended path to the file after the move
* @return A 0 return value indicates success. A -1 return value indicates an
* error, and an error code is stored in errno
*/
int
vlc_rename
(
const
char
*
oldpath
,
const
char
*
newpath
)
{
#if defined (WIN32)
CONVERT_PATH
(
oldpath
,
wold
,
-
1
);
CONVERT_PATH
(
newpath
,
wnew
,
-
1
);
# ifdef UNDER_CE
/* FIXME: errno support */
if
(
MoveFileW
(
wold
,
wnew
))
return
0
;
else
return
-
1
;
#else
if
(
_wrename
(
wold
,
wnew
)
&&
errno
==
EACCES
)
{
/* Windows does not allow atomic file replacement */
if
(
_wremove
(
wnew
))
{
errno
=
EACCES
;
/* restore errno */
return
-
1
;
}
if
(
_wrename
(
wold
,
wnew
))
return
-
1
;
}
return
0
;
#endif
#endif
const
char
*
lo
=
ToLocale
(
oldpath
);
if
(
lo
==
NULL
)
goto
error
;
const
char
*
ln
=
ToLocale
(
newpath
);
if
(
ln
==
NULL
)
{
LocaleFree
(
lo
);
error:
errno
=
ENOENT
;
return
-
1
;
}
int
ret
=
rename
(
lo
,
ln
);
LocaleFree
(
lo
);
LocaleFree
(
ln
);
return
ret
;
}
int
vlc_mkstemp
(
char
*
template
)
{
static
const
char
digits
[]
=
"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"
;
...
...
@@ -623,130 +234,3 @@ int vlc_mkstemp( char *template )
errno
=
EEXIST
;
return
-
1
;
}
#ifdef UNDER_CE
# define dup(fd) (fd, -1)
#endif
/**
* Duplicates a file descriptor. The new file descriptor has the close-on-exec
* descriptor flag set.
* @return a new file descriptor or -1
*/
int
vlc_dup
(
int
oldfd
)
{
int
newfd
;
#ifdef HAVE_DUP3
/* Unfortunately, dup3() works like dup2(), not like plain dup(). So we
* need such contortion to find the new file descriptor while preserving
* thread safety of the file descriptor table. */
newfd
=
vlc_open
(
"/dev/null"
,
O_RDONLY
);
if
(
likely
(
newfd
!=
-
1
))
{
if
(
likely
(
dup3
(
oldfd
,
newfd
,
O_CLOEXEC
)
==
newfd
))
return
newfd
;
close
(
newfd
);
}
#endif
newfd
=
dup
(
oldfd
);
#ifdef HAVE_FCNTL
if
(
likely
(
newfd
!=
-
1
))
fcntl
(
newfd
,
F_SETFD
,
FD_CLOEXEC
);
#endif
return
newfd
;
}
#include <vlc_network.h>
/**
* Creates a socket file descriptor. The new file descriptor has the
* close-on-exec flag set.
* @param pf protocol family
* @param type socket type
* @param proto network protocol
* @param nonblock true to create a non-blocking socket
* @return a new file descriptor or -1
*/
int
vlc_socket
(
int
pf
,
int
type
,
int
proto
,
bool
nonblock
)
{
int
fd
;
#ifdef SOCK_CLOEXEC
type
|=
SOCK_CLOEXEC
;
if
(
nonblock
)
type
|=
SOCK_NONBLOCK
;
fd
=
socket
(
pf
,
type
,
proto
);
if
(
fd
!=
-
1
||
errno
!=
EINVAL
)
return
fd
;
type
&=
~
(
SOCK_CLOEXEC
|
SOCK_NONBLOCK
);
#endif
fd
=
socket
(
pf
,
type
,
proto
);
if
(
fd
==
-
1
)
return
-
1
;
#ifndef WIN32
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
if
(
nonblock
)
fcntl
(
fd
,
F_SETFL
,
fcntl
(
fd
,
F_GETFL
,
0
)
|
O_NONBLOCK
);
#else
if
(
nonblock
)
ioctlsocket
(
fd
,
FIONBIO
,
&
(
unsigned
long
){
1
});
#endif
return
fd
;
}
/**
* Accepts an inbound connection request on a listening socket.
* The new file descriptor has the close-on-exec flag set.
* @param lfd listening socket file descriptor
* @param addr pointer to the peer address or NULL [OUT]
* @param alen pointer to the length of the peer address or NULL [OUT]
* @param nonblock whether to put the new socket in non-blocking mode
* @return a new file descriptor, or -1 on error.
*/
int
vlc_accept
(
int
lfd
,
struct
sockaddr
*
addr
,
socklen_t
*
alen
,
bool
nonblock
)
{
#ifdef HAVE_ACCEPT4
int
flags
=
SOCK_CLOEXEC
;
if
(
nonblock
)
flags
|=
SOCK_NONBLOCK
;
do
{
int
fd
=
accept4
(
lfd
,
addr
,
alen
,
flags
);
if
(
fd
!=
-
1
)
return
fd
;
}
while
(
errno
==
EINTR
);
if
(
errno
!=
ENOSYS
)
return
-
1
;
#endif
#ifdef WIN32
errno
=
0
;
#endif
do
{
int
fd
=
accept
(
lfd
,
addr
,
alen
);
if
(
fd
!=
-
1
)
{
#ifndef WIN32
fcntl
(
fd
,
F_SETFD
,
FD_CLOEXEC
);
if
(
nonblock
)
fcntl
(
fd
,
F_SETFL
,
fcntl
(
fd
,
F_GETFL
,
0
)
|
O_NONBLOCK
);
#else
if
(
nonblock
)
ioctlsocket
(
fd
,
FIONBIO
,
&
(
unsigned
long
){
1
});
#endif
return
fd
;
}
}
while
(
errno
==
EINTR
);
return
-
1
;
}
src/win32/filesystem.c
0 → 100644
View file @
485d0ebd
/*****************************************************************************
* filesystem.c: Windows file system helpers
*****************************************************************************
* Copyright (C) 2005-2006 the VideoLAN team
* Copyright © 2005-2008 Rémi Denis-Courmont
*
* Authors: Rémi Denis-Courmont <rem # videolan.org>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
*****************************************************************************/
/*****************************************************************************
* Preamble
*****************************************************************************/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <vlc_common.h>
#include <vlc_charset.h>
#include <vlc_fs.h>
#include "libvlc.h"
/* vlc_mkdir */
#include <assert.h>
#include <stdio.h>
#include <errno.h>
#include <sys/types.h>
#include <dirent.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <winsock2.h>
#ifndef UNDER_CE
# include <direct.h>
#else
# include <tchar.h>
#endif
static
int
convert_path
(
const
char
*
restrict
path
,
wchar_t
*
restrict
wpath
)
{
if
(
!
MultiByteToWideChar
(
CP_UTF8
,
0
,
path
,
-
1
,
wpath
,
MAX_PATH
))
{
errno
=
ENOENT
;
return
-
1
;
}
wpath
[
MAX_PATH
]
=
L'\0'
;
return
0
;
}
#define CONVERT_PATH(path, wpath, err) \
wchar_t wpath[MAX_PATH+1]; \
if (convert_path (path, wpath)) \
return (err)
int
vlc_open
(
const
char
*
filename
,
int
flags
,
...)
{
unsigned
int
mode
=
0
;
va_list
ap
;
va_start
(
ap
,
flags
);
if
(
flags
&
O_CREAT
)
mode
=
va_arg
(
ap
,
unsigned
int
);
va_end
(
ap
);
#ifdef UNDER_CE
/*_open translates to wchar internally on WinCE*/
return
_open
(
filename
,
flags
,
mode
);
#else
/*
* open() cannot open files with non-“ANSI” characters on Windows.
* We use _wopen() instead. Same thing for mkdir() and stat().
*/
CONVERT_PATH
(
filename
,
wpath
,
-
1
);
return
_wopen
(
wpath
,
flags
,
mode
);
#endif
}
int
vlc_openat
(
int
dir
,
const
char
*
filename
,
int
flags
,
...)
{
(
void
)
dir
;
(
void
)
filename
;
(
void
)
flags
;
errno
=
ENOSYS
;
return
-
1
;
}
int
vlc_mkdir
(
const
char
*
dirname
,
mode_t
mode
)
{
#if defined (UNDER_CE)
(
void
)
mode
;
/* mkdir converts internally to wchar */
return
_mkdir
(
dirname
);
#else
(
void
)
mode
;
CONVERT_PATH
(
dirname
,
wpath
,
-
1
);
return
_wmkdir
(
wpath
);
#endif
}
DIR
*
vlc_opendir
(
const
char
*
dirname
)
{
CONVERT_PATH
(
dirname
,
wpath
,
NULL
);
return
(
DIR
*
)
vlc_wopendir
(
wpath
);
}
char
*
vlc_readdir
(
DIR
*
dir
)
{
struct
_wdirent
*
ent
=
vlc_wreaddir
(
dir
);
if
(
ent
==
NULL
)
return
NULL
;
return
FromWide
(
ent
->
d_name
);
}
int
vlc_stat
(
const
char
*
filename
,
struct
stat
*
buf
)
{
#ifdef UNDER_CE
/* _stat translates to wchar internally on WinCE */
return
_stat
(
filename
,
buf
);
#else
CONVERT_PATH
(
filename
,
wpath
,
-
1
);
return
_wstati64
(
wpath
,
buf
);
#endif
}
int
vlc_lstat
(
const
char
*
filename
,
struct
stat
*
buf
)
{
return
vlc_stat
(
filename
,
buf
);
}
int
vlc_unlink
(
const
char
*
filename
)
{
#ifdef UNDER_CE
/*_open translates to wchar internally on WinCE*/
return
_unlink
(
filename
);
#else
CONVERT_PATH
(
filename
,
wpath
,
-
1
);
return
_wunlink
(
wpath
);
#endif
}
int
vlc_rename
(
const
char
*
oldpath
,
const
char
*
newpath
)
{
CONVERT_PATH
(
oldpath
,
wold
,
-
1
);
CONVERT_PATH
(
newpath
,
wnew
,
-
1
);
# ifdef UNDER_CE
/* FIXME: errno support */
return
MoveFileW
(
wold
,
wnew
)
?
0
:
-
1
;
#else
if
(
_wrename
(
wold
,
wnew
)
&&
errno
==
EACCES
)
{
/* Windows does not allow atomic file replacement */
if
(
_wremove
(
wnew
))
{
errno
=
EACCES
;
/* restore errno */
return
-
1
;
}
if
(
_wrename
(
wold
,
wnew
))
return
-
1
;
}
return
0
;
#endif
}
int
vlc_dup
(
int
oldfd
)
{
#ifdef UNDER_CE
(
void
)
oldfd
;
errno
=
ENOSYS
;
return
-
1
;
#else
return
dup
(
oldfd
);
#endif
}
#include <vlc_network.h>
int
vlc_socket
(
int
pf
,
int
type
,
int
proto
,
bool
nonblock
)
{
int
fd
=
socket
(
pf
,
type
,
proto
);
if
(
fd
==
-
1
)
return
-
1
;
if
(
nonblock
)
ioctlsocket
(
fd
,
FIONBIO
,
&
(
unsigned
long
){
1
});
return
fd
;
}
int
vlc_accept
(
int
lfd
,
struct
sockaddr
*
addr
,
socklen_t
*
alen
,
bool
nonblock
)
{
int
fd
=
accept
(
lfd
,
addr
,
alen
);
if
(
fd
!=
-
1
&&
nonblock
)
ioctlsocket
(
fd
,
FIONBIO
,
&
(
unsigned
long
){
1
});
return
fd
;
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment