121 lines
2.7 KiB
C
121 lines
2.7 KiB
C
|
#define __CRT__NO_INLINE
|
||
|
#include <sys/stat.h>
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
/**
|
||
|
* Returns _path without trailing slash if any
|
||
|
*
|
||
|
* - if _path has no trailing slash, the function returns it
|
||
|
* - if _path has a trailing slash, but is of the form C:/, then it returns it
|
||
|
* - otherwise, the function creates a new string, which is a copy of _path
|
||
|
* without the trailing slash. It is then the responsibility of the caller
|
||
|
* to free it.
|
||
|
*/
|
||
|
|
||
|
static char*
|
||
|
_mingw_no_trailing_slash (const char* _path)
|
||
|
{
|
||
|
int len;
|
||
|
char *p;
|
||
|
|
||
|
p = (char*)_path;
|
||
|
|
||
|
if (_path && *_path) {
|
||
|
len = strlen (_path);
|
||
|
|
||
|
/* Ignore X:\ */
|
||
|
|
||
|
if (len <= 1 || ((len == 2 || len == 3) && _path[1] == ':'))
|
||
|
return p;
|
||
|
|
||
|
/* Check UNC \\abc\<name>\ */
|
||
|
if ((_path[0] == '\\' || _path[0] == '/')
|
||
|
&& (_path[1] == '\\' || _path[1] == '/'))
|
||
|
{
|
||
|
const char *r = &_path[2];
|
||
|
while (*r != 0 && *r != '\\' && *r != '/')
|
||
|
++r;
|
||
|
if (*r != 0)
|
||
|
++r;
|
||
|
if (*r == 0)
|
||
|
return p;
|
||
|
while (*r != 0 && *r != '\\' && *r != '/')
|
||
|
++r;
|
||
|
if (*r != 0)
|
||
|
++r;
|
||
|
if (*r == 0)
|
||
|
return p;
|
||
|
}
|
||
|
|
||
|
if (_path[len - 1] == '/' || _path[len - 1] == '\\')
|
||
|
{
|
||
|
p = (char*)malloc (len);
|
||
|
memcpy (p, _path, len - 1);
|
||
|
p[len - 1] = '\0';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return p;
|
||
|
}
|
||
|
/* FIXME: Relying on _USE_32BIT_TIME_T, which is a user-macro,
|
||
|
during CRT compilation is plainly broken. Need an appropriate
|
||
|
implementation to provide users the ability of compiling the
|
||
|
CRT only with 32-bit time_t behavior. */
|
||
|
#if defined(_USE_32BIT_TIME_T)
|
||
|
int __cdecl
|
||
|
stat(const char *_Filename,struct stat *_Stat)
|
||
|
{
|
||
|
struct _stat32 st;
|
||
|
char *_path = _mingw_no_trailing_slash(_Filename);
|
||
|
|
||
|
int ret=_stat32(_path,&st);
|
||
|
|
||
|
if (_path != _Filename)
|
||
|
free (_path);
|
||
|
|
||
|
if (ret == -1) {
|
||
|
memset(_Stat,0,sizeof(struct stat));
|
||
|
return -1;
|
||
|
}
|
||
|
/* struct stat and struct _stat32
|
||
|
are the same for this case. */
|
||
|
memcpy(_Stat, &st, sizeof(struct _stat32));
|
||
|
return ret;
|
||
|
}
|
||
|
#else
|
||
|
int __cdecl
|
||
|
stat(const char *_Filename,struct stat *_Stat)
|
||
|
{
|
||
|
struct _stat64 st;
|
||
|
char *_path = _mingw_no_trailing_slash(_Filename);
|
||
|
|
||
|
int ret=_stat64(_path,&st);
|
||
|
|
||
|
if (_path != _Filename)
|
||
|
free (_path);
|
||
|
|
||
|
if (ret == -1) {
|
||
|
memset(_Stat,0,sizeof(struct stat));
|
||
|
return -1;
|
||
|
}
|
||
|
/* struct stat and struct _stat64i32
|
||
|
are the same for this case. */
|
||
|
_Stat->st_dev=st.st_dev;
|
||
|
_Stat->st_ino=st.st_ino;
|
||
|
_Stat->st_mode=st.st_mode;
|
||
|
_Stat->st_nlink=st.st_nlink;
|
||
|
_Stat->st_uid=st.st_uid;
|
||
|
_Stat->st_gid=st.st_gid;
|
||
|
_Stat->st_rdev=st.st_rdev;
|
||
|
_Stat->st_size=(_off_t) st.st_size;
|
||
|
_Stat->st_atime=st.st_atime;
|
||
|
_Stat->st_mtime=st.st_mtime;
|
||
|
_Stat->st_ctime=st.st_ctime;
|
||
|
return ret;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/* Add __imp__fstat and __imp__stat symbols. */
|
||
|
int (*__MINGW_IMP_SYMBOL(stat))(const char *,struct stat *) = &stat;
|
||
|
|