| #define __CRT__NO_INLINE |
| #include <sys/stat.h> |
| #include <stdlib.h> |
| #include <malloc.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 wchar_t* |
| _mingw_no_trailing_slash (const wchar_t* _path) |
| { |
| int len; |
| wchar_t *p; |
| |
| p = (wchar_t*)_path; |
| |
| if (_path && *_path) { |
| len = wcslen (_path); |
| |
| /* Ignore X:\ */ |
| |
| if (len <= 1 || ((len == 2 || len == 3) && _path[1] == L':')) |
| return p; |
| |
| /* Check UNC \\abc\<name>\ */ |
| if ((_path[0] == L'\\' || _path[0] == L'/') |
| && (_path[1] == L'\\' || _path[1] == L'/')) |
| { |
| const wchar_t *r = &_path[2]; |
| while (*r != 0 && *r != L'\\' && *r != L'/') |
| ++r; |
| if (*r != 0) |
| ++r; |
| if (*r == 0) |
| return p; |
| while (*r != 0 && *r != L'\\' && *r != L'/') |
| ++r; |
| if (*r != 0) |
| ++r; |
| if (*r == 0) |
| return p; |
| } |
| |
| if (_path[len - 1] == L'/' || _path[len - 1] == L'\\') |
| { |
| p = (wchar_t*)malloc (len * sizeof(wchar_t)); |
| memcpy (p, _path, (len - 1) * sizeof(wchar_t)); |
| p[len - 1] = L'\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 |
| wstat(const wchar_t *_Filename,struct stat *_Stat) |
| { |
| struct _stat32 st; |
| wchar_t *_path = _mingw_no_trailing_slash(_Filename); |
| |
| int ret=_wstat32(_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 |
| wstat(const wchar_t *_Filename,struct stat *_Stat) |
| { |
| struct _stat64 st; |
| wchar_t *_path = _mingw_no_trailing_slash(_Filename); |
| |
| int ret=_wstat64(_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 |
| |