#define __CRT__NO_INLINE #include #include #include /** * 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\\ */ 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