|  | #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; | 
|  |  |