2021-07-28 21:11:07 +01:00
|
|
|
#include "fs.h"
|
|
|
|
|
2021-12-21 17:39:55 +00:00
|
|
|
#if MG_ENABLE_FILE
|
2021-09-16 11:16:10 +01:00
|
|
|
static int p_stat(const char *path, size_t *size, time_t *mtime) {
|
2021-07-28 21:11:07 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
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
|
|
|
|
struct stat st;
|
|
|
|
if (stat(path, &st) != 0) return 0;
|
|
|
|
#endif
|
|
|
|
if (size) *size = (size_t) st.st_size;
|
2021-07-29 14:21:20 +01:00
|
|
|
if (mtime) *mtime = st.st_mtime;
|
2021-07-28 21:11:07 +01:00
|
|
|
return MG_FS_READ | MG_FS_WRITE | (S_ISDIR(st.st_mode) ? MG_FS_DIR : 0);
|
|
|
|
}
|
|
|
|
|
2021-07-29 14:21:20 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
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 11:16:10 +01:00
|
|
|
static void p_list(const char *dir, void (*fn)(const char *, void *),
|
|
|
|
void *userdata) {
|
2021-07-29 14:21:20 +01:00
|
|
|
#if MG_ENABLE_DIRLIST
|
2021-07-28 21:11:07 +01:00
|
|
|
struct dirent *dp;
|
|
|
|
DIR *dirp;
|
2021-07-29 14:21:20 +01: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-28 21:11:07 +01:00
|
|
|
}
|
2021-07-29 14:21:20 +01:00
|
|
|
closedir(dirp);
|
|
|
|
#else
|
|
|
|
(void) dir, (void) fn, (void) userdata;
|
|
|
|
#endif
|
2021-07-28 21:11:07 +01:00
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static struct mg_fd *p_open(const char *path, int flags) {
|
2021-09-02 10:20:55 +01:00
|
|
|
const char *mode = flags == (MG_FS_READ | MG_FS_WRITE) ? "r+b"
|
|
|
|
: flags & MG_FS_READ ? "rb"
|
|
|
|
: flags & MG_FS_WRITE ? "wb"
|
|
|
|
: "";
|
2021-07-28 21:11:07 +01:00
|
|
|
void *fp = NULL;
|
|
|
|
struct mg_fd *fd = NULL;
|
|
|
|
#ifdef _WIN32
|
|
|
|
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]));
|
|
|
|
fp = (void *) _wfopen(b1, b2);
|
|
|
|
#else
|
|
|
|
fp = (void *) fopen(path, mode);
|
|
|
|
#endif
|
|
|
|
if (fp == NULL) return NULL;
|
2021-07-29 14:21:20 +01:00
|
|
|
fd = (struct mg_fd *) calloc(1, sizeof(*fd));
|
2021-07-28 21:11:07 +01:00
|
|
|
fd->fd = fp;
|
|
|
|
fd->fs = &mg_fs_posix;
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static void p_close(struct mg_fd *fd) {
|
2021-07-29 14:21:20 +01:00
|
|
|
if (fd != NULL) fclose((FILE *) fd->fd), free(fd);
|
2021-07-28 21:11:07 +01:00
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static size_t p_read(void *fp, void *buf, size_t len) {
|
2021-07-28 21:11:07 +01:00
|
|
|
return fread(buf, 1, len, (FILE *) fp);
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static size_t p_write(void *fp, const void *buf, size_t len) {
|
2021-07-28 21:11:07 +01:00
|
|
|
return fwrite(buf, 1, len, (FILE *) fp);
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static size_t p_seek(void *fp, size_t offset) {
|
2021-12-21 17:39:55 +00:00
|
|
|
#if (defined(_FILE_OFFSET_BITS) && _FILE_OFFSET_BITS == 64) || \
|
|
|
|
(defined(_POSIX_C_SOURCE) && _POSIX_C_SOURCE >= 200112L) || \
|
|
|
|
(defined(_XOPEN_SOURCE) && _XOPEN_SOURCE >= 600)
|
2021-07-28 21:11:07 +01:00
|
|
|
fseeko((FILE *) fp, (off_t) offset, SEEK_SET);
|
|
|
|
#else
|
|
|
|
fseek((FILE *) fp, (long) offset, SEEK_SET);
|
|
|
|
#endif
|
|
|
|
return (size_t) ftell((FILE *) fp);
|
|
|
|
}
|
2021-12-21 18:16:12 +00:00
|
|
|
|
2021-07-28 21:11:07 +01:00
|
|
|
#else
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static int p_stat(const char *path, size_t *size, time_t *mtime) {
|
2021-07-28 21:11:07 +01:00
|
|
|
(void) path, (void) size, (void) mtime;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static void p_list(const char *path, void (*fn)(const char *, void *),
|
|
|
|
void *userdata) {
|
2021-07-28 21:11:07 +01:00
|
|
|
(void) path, (void) fn, (void) userdata;
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static struct mg_fd *p_open(const char *path, int flags) {
|
2021-07-28 21:11:07 +01:00
|
|
|
(void) path, (void) flags;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static void p_close(struct mg_fd *fd) {
|
2021-07-28 21:11:07 +01:00
|
|
|
(void) fd;
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static size_t p_read(void *fd, void *buf, size_t len) {
|
2021-07-28 21:11:07 +01:00
|
|
|
(void) fd, (void) buf, (void) len;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static size_t p_write(void *fd, const void *buf, size_t len) {
|
2021-07-28 21:11:07 +01:00
|
|
|
(void) fd, (void) buf, (void) len;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
static size_t p_seek(void *fd, size_t offset) {
|
2021-07-28 21:11:07 +01:00
|
|
|
(void) fd, (void) offset;
|
|
|
|
return (size_t) ~0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-09-16 11:16:10 +01:00
|
|
|
struct mg_fs mg_fs_posix = {p_stat, p_list, p_open, p_close,
|
|
|
|
p_read, p_write, p_seek};
|