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