2021-07-29 04:11:07 +08:00
|
|
|
#include "fs.h"
|
|
|
|
|
2021-12-22 01:39:55 +08:00
|
|
|
#if MG_ENABLE_FILE
|
2022-01-11 02:07:45 +08:00
|
|
|
|
|
|
|
#ifndef MG_STAT_STRUCT
|
|
|
|
#define MG_STAT_STRUCT stat
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef MG_STAT_FUNC
|
|
|
|
#define MG_STAT_FUNC stat
|
|
|
|
#endif
|
|
|
|
|
2021-09-16 18:16:10 +08:00
|
|
|
static int p_stat(const char *path, size_t *size, time_t *mtime) {
|
2021-12-23 02:04:16 +08:00
|
|
|
#if !defined(S_ISDIR)
|
2022-02-13 02:17:25 +08:00
|
|
|
MG_ERROR(("stat() API is not supported. %p %p %p", path, size, mtime));
|
2021-12-23 02:04:16 +08:00
|
|
|
return 0;
|
|
|
|
#else
|
2022-02-12 22:33:43 +08:00
|
|
|
#if MG_ARCH == MG_ARCH_WIN32
|
2021-07-29 04:11:07 +08:00
|
|
|
struct _stati64 st;
|
|
|
|
wchar_t tmp[PATH_MAX];
|
|
|
|
MultiByteToWideChar(CP_UTF8, 0, path, -1, tmp, sizeof(tmp) / sizeof(tmp[0]));
|
|
|
|
if (_wstati64(tmp, &st) != 0) return 0;
|
|
|
|
#else
|
2022-01-11 02:07:45 +08:00
|
|
|
struct MG_STAT_STRUCT st;
|
|
|
|
if (MG_STAT_FUNC(path, &st) != 0) return 0;
|
2021-07-29 04:11:07 +08:00
|
|
|
#endif
|
|
|
|
if (size) *size = (size_t) st.st_size;
|
2021-07-29 21:21:20 +08:00
|
|
|
if (mtime) *mtime = st.st_mtime;
|
2021-07-29 04:11:07 +08:00
|
|
|
return MG_FS_READ | MG_FS_WRITE | (S_ISDIR(st.st_mode) ? MG_FS_DIR : 0);
|
2021-12-23 02:04:16 +08:00
|
|
|
#endif
|
2021-07-29 04:11:07 +08:00
|
|
|
}
|
|
|
|
|
2022-02-12 22:33:43 +08:00
|
|
|
#if MG_ARCH == MG_ARCH_WIN32
|
2021-07-29 21:21:20 +08:00
|
|
|
struct dirent {
|
|
|
|
char d_name[MAX_PATH];
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct win32_dir {
|
|
|
|
HANDLE handle;
|
|
|
|
WIN32_FIND_DATAW info;
|
|
|
|
struct dirent result;
|
|
|
|
} DIR;
|
|
|
|
|
|
|
|
int gettimeofday(struct timeval *tv, void *tz) {
|
|
|
|
FILETIME ft;
|
|
|
|
unsigned __int64 tmpres = 0;
|
|
|
|
|
|
|
|
if (tv != NULL) {
|
|
|
|
GetSystemTimeAsFileTime(&ft);
|
|
|
|
tmpres |= ft.dwHighDateTime;
|
|
|
|
tmpres <<= 32;
|
|
|
|
tmpres |= ft.dwLowDateTime;
|
|
|
|
tmpres /= 10; // convert into microseconds
|
|
|
|
tmpres -= (int64_t) 11644473600000000;
|
|
|
|
tv->tv_sec = (long) (tmpres / 1000000UL);
|
|
|
|
tv->tv_usec = (long) (tmpres % 1000000UL);
|
|
|
|
}
|
|
|
|
(void) tz;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int to_wchar(const char *path, wchar_t *wbuf, size_t wbuf_len) {
|
|
|
|
int ret;
|
|
|
|
char buf[MAX_PATH * 2], buf2[MAX_PATH * 2], *p;
|
|
|
|
strncpy(buf, path, sizeof(buf));
|
|
|
|
buf[sizeof(buf) - 1] = '\0';
|
|
|
|
// Trim trailing slashes. Leave backslash for paths like "X:\"
|
|
|
|
p = buf + strlen(buf) - 1;
|
|
|
|
while (p > buf && p[-1] != ':' && (p[0] == '\\' || p[0] == '/')) *p-- = '\0';
|
|
|
|
memset(wbuf, 0, wbuf_len * sizeof(wchar_t));
|
|
|
|
ret = MultiByteToWideChar(CP_UTF8, 0, buf, -1, wbuf, (int) wbuf_len);
|
|
|
|
// Convert back to Unicode. If doubly-converted string does not match the
|
|
|
|
// original, something is fishy, reject.
|
|
|
|
WideCharToMultiByte(CP_UTF8, 0, wbuf, (int) wbuf_len, buf2, sizeof(buf2),
|
|
|
|
NULL, NULL);
|
|
|
|
if (strcmp(buf, buf2) != 0) {
|
|
|
|
wbuf[0] = L'\0';
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
DIR *opendir(const char *name) {
|
|
|
|
DIR *d = NULL;
|
|
|
|
wchar_t wpath[MAX_PATH];
|
|
|
|
DWORD attrs;
|
|
|
|
|
|
|
|
if (name == NULL) {
|
|
|
|
SetLastError(ERROR_BAD_ARGUMENTS);
|
|
|
|
} else if ((d = (DIR *) calloc(1, sizeof(*d))) == NULL) {
|
|
|
|
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
|
|
|
} else {
|
|
|
|
to_wchar(name, wpath, sizeof(wpath) / sizeof(wpath[0]));
|
|
|
|
attrs = GetFileAttributesW(wpath);
|
|
|
|
if (attrs != 0Xffffffff && (attrs & FILE_ATTRIBUTE_DIRECTORY)) {
|
|
|
|
(void) wcscat(wpath, L"\\*");
|
|
|
|
d->handle = FindFirstFileW(wpath, &d->info);
|
|
|
|
d->result.d_name[0] = '\0';
|
|
|
|
} else {
|
|
|
|
free(d);
|
|
|
|
d = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
int closedir(DIR *d) {
|
|
|
|
int result = 0;
|
|
|
|
if (d != NULL) {
|
|
|
|
if (d->handle != INVALID_HANDLE_VALUE)
|
|
|
|
result = FindClose(d->handle) ? 0 : -1;
|
|
|
|
free(d);
|
|
|
|
} else {
|
|
|
|
result = -1;
|
|
|
|
SetLastError(ERROR_BAD_ARGUMENTS);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct dirent *readdir(DIR *d) {
|
|
|
|
struct dirent *result = NULL;
|
|
|
|
if (d != NULL) {
|
|
|
|
memset(&d->result, 0, sizeof(d->result));
|
|
|
|
if (d->handle != INVALID_HANDLE_VALUE) {
|
|
|
|
result = &d->result;
|
|
|
|
WideCharToMultiByte(CP_UTF8, 0, d->info.cFileName, -1, result->d_name,
|
|
|
|
sizeof(result->d_name), NULL, NULL);
|
|
|
|
if (!FindNextFileW(d->handle, &d->info)) {
|
|
|
|
FindClose(d->handle);
|
|
|
|
d->handle = INVALID_HANDLE_VALUE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SetLastError(ERROR_FILE_NOT_FOUND);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SetLastError(ERROR_BAD_ARGUMENTS);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-09-16 18:16:10 +08:00
|
|
|
static void p_list(const char *dir, void (*fn)(const char *, void *),
|
|
|
|
void *userdata) {
|
2021-07-29 21:21:20 +08:00
|
|
|
#if MG_ENABLE_DIRLIST
|
2021-07-29 04:11:07 +08:00
|
|
|
struct dirent *dp;
|
|
|
|
DIR *dirp;
|
2021-07-29 21:21:20 +08:00
|
|
|
if ((dirp = (opendir(dir))) == NULL) return;
|
|
|
|
while ((dp = readdir(dirp)) != NULL) {
|
|
|
|
if (!strcmp(dp->d_name, ".") || !strcmp(dp->d_name, "..")) continue;
|
|
|
|
fn(dp->d_name, userdata);
|
2021-07-29 04:11:07 +08:00
|
|
|
}
|
2021-07-29 21:21:20 +08:00
|
|
|
closedir(dirp);
|
|
|
|
#else
|
|
|
|
(void) dir, (void) fn, (void) userdata;
|
|
|
|
#endif
|
2021-07-29 04:11:07 +08:00
|
|
|
}
|
|
|
|
|
2022-01-17 22:42:41 +08:00
|
|
|
static void *p_open(const char *path, int flags) {
|
2022-01-19 01:11:02 +08:00
|
|
|
const char *mode = flags == MG_FS_READ ? "rb" : "a+b";
|
2022-02-12 22:33:43 +08:00
|
|
|
#if MG_ARCH == MG_ARCH_WIN32
|
2021-07-29 04:11:07 +08:00
|
|
|
wchar_t b1[PATH_MAX], b2[10];
|
|
|
|
MultiByteToWideChar(CP_UTF8, 0, path, -1, b1, sizeof(b1) / sizeof(b1[0]));
|
|
|
|
MultiByteToWideChar(CP_UTF8, 0, mode, -1, b2, sizeof(b2) / sizeof(b2[0]));
|
2022-01-17 22:42:41 +08:00
|
|
|
return (void *) _wfopen(b1, b2);
|
2021-07-29 04:11:07 +08:00
|
|
|
#else
|
2022-01-17 22:42:41 +08:00
|
|
|
return (void *) fopen(path, mode);
|
2021-07-29 04:11:07 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2022-01-17 22:42:41 +08:00
|
|
|
static void p_close(void *fp) {
|
2022-01-19 01:11:02 +08:00
|
|
|
fclose((FILE *) fp);
|
2021-07-29 04:11:07 +08:00
|
|
|
}
|
|
|
|
|
2021-09-16 18:16:10 +08:00
|
|
|
static size_t p_read(void *fp, void *buf, size_t len) {
|
2021-07-29 04:11:07 +08:00
|
|
|
return fread(buf, 1, len, (FILE *) fp);
|
|
|
|
}
|
|
|
|
|
2021-09-16 18:16:10 +08:00
|
|
|
static size_t p_write(void *fp, const void *buf, size_t len) {
|
2021-07-29 04:11:07 +08:00
|
|
|
return fwrite(buf, 1, len, (FILE *) fp);
|
|
|
|
}
|
|
|
|
|
2021-09-16 18:16:10 +08:00
|
|
|
static size_t p_seek(void *fp, size_t offset) {
|
2021-12-22 01:39:55 +08:00
|
|
|
#if (defined(_FILE_OFFSET_BITS) && _FILE_OFFSET_BITS == 64) || \
|
|
|
|
(defined(_POSIX_C_SOURCE) && _POSIX_C_SOURCE >= 200112L) || \
|
|
|
|
(defined(_XOPEN_SOURCE) && _XOPEN_SOURCE >= 600)
|
2022-02-23 21:30:14 +08:00
|
|
|
if (fseeko((FILE *) fp, (off_t) offset, SEEK_SET) != 0) (void) 0;
|
2021-07-29 04:11:07 +08:00
|
|
|
#else
|
2022-02-23 21:30:14 +08:00
|
|
|
if (fseek((FILE *) fp, (long) offset, SEEK_SET) != 0) (void) 0;
|
2021-07-29 04:11:07 +08:00
|
|
|
#endif
|
|
|
|
return (size_t) ftell((FILE *) fp);
|
|
|
|
}
|
2021-12-22 02:16:12 +08:00
|
|
|
|
2022-01-19 01:11:02 +08:00
|
|
|
static bool p_rename(const char *from, const char *to) {
|
|
|
|
return rename(from, to) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool p_remove(const char *path) {
|
|
|
|
return remove(path) == 0;
|
|
|
|
}
|
|
|
|
|
2022-01-19 03:19:34 +08:00
|
|
|
static bool p_mkdir(const char *path) {
|
|
|
|
return mkdir(path, 0775) == 0;
|
|
|
|
}
|
|
|
|
|
2021-07-29 04:11:07 +08:00
|
|
|
#else
|
|
|
|
|
2021-09-16 18:16:10 +08:00
|
|
|
static int p_stat(const char *path, size_t *size, time_t *mtime) {
|
2021-07-29 04:11:07 +08:00
|
|
|
(void) path, (void) size, (void) mtime;
|
|
|
|
return 0;
|
|
|
|
}
|
2021-09-16 18:16:10 +08:00
|
|
|
static void p_list(const char *path, void (*fn)(const char *, void *),
|
|
|
|
void *userdata) {
|
2021-07-29 04:11:07 +08:00
|
|
|
(void) path, (void) fn, (void) userdata;
|
|
|
|
}
|
2022-01-19 03:19:34 +08:00
|
|
|
static void *p_open(const char *path, int flags) {
|
2021-07-29 04:11:07 +08:00
|
|
|
(void) path, (void) flags;
|
|
|
|
return NULL;
|
|
|
|
}
|
2022-01-19 03:19:34 +08:00
|
|
|
static void p_close(void *fp) {
|
|
|
|
(void) fp;
|
2021-07-29 04:11:07 +08:00
|
|
|
}
|
2021-09-16 18:16:10 +08:00
|
|
|
static size_t p_read(void *fd, void *buf, size_t len) {
|
2021-07-29 04:11:07 +08:00
|
|
|
(void) fd, (void) buf, (void) len;
|
|
|
|
return 0;
|
|
|
|
}
|
2021-09-16 18:16:10 +08:00
|
|
|
static size_t p_write(void *fd, const void *buf, size_t len) {
|
2021-07-29 04:11:07 +08:00
|
|
|
(void) fd, (void) buf, (void) len;
|
|
|
|
return 0;
|
|
|
|
}
|
2021-09-16 18:16:10 +08:00
|
|
|
static size_t p_seek(void *fd, size_t offset) {
|
2021-07-29 04:11:07 +08:00
|
|
|
(void) fd, (void) offset;
|
|
|
|
return (size_t) ~0;
|
|
|
|
}
|
2022-01-19 01:11:02 +08:00
|
|
|
static bool p_rename(const char *from, const char *to) {
|
|
|
|
(void) from, (void) to;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
static bool p_remove(const char *path) {
|
|
|
|
(void) path;
|
|
|
|
return false;
|
|
|
|
}
|
2022-01-19 03:19:34 +08:00
|
|
|
static bool p_mkdir(const char *path) {
|
|
|
|
(void) path;
|
|
|
|
return false;
|
|
|
|
}
|
2021-07-29 04:11:07 +08:00
|
|
|
#endif
|
|
|
|
|
2022-01-19 03:19:34 +08:00
|
|
|
struct mg_fs mg_fs_posix = {p_stat, p_list, p_open, p_close, p_read,
|
|
|
|
p_write, p_seek, p_rename, p_remove, p_mkdir};
|