2021-07-28 21:11:07 +01:00
|
|
|
#include "fs.h"
|
2023-03-27 14:11:39 -03:00
|
|
|
#include "printf.h"
|
2022-02-10 11:56:55 +00:00
|
|
|
#include "str.h"
|
2021-07-28 21:11:07 +01:00
|
|
|
|
|
|
|
struct packed_file {
|
|
|
|
const char *data;
|
|
|
|
size_t size;
|
|
|
|
size_t pos;
|
|
|
|
};
|
|
|
|
|
2022-05-13 22:15:33 +01:00
|
|
|
const char *mg_unpack(const char *path, size_t *size, time_t *mtime);
|
|
|
|
const char *mg_unlist(size_t no);
|
|
|
|
|
2021-07-30 13:19:20 +01:00
|
|
|
#if MG_ENABLE_PACKED_FS
|
|
|
|
#else
|
|
|
|
const char *mg_unpack(const char *path, size_t *size, time_t *mtime) {
|
|
|
|
(void) path, (void) size, (void) mtime;
|
2021-07-28 21:11:07 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
2021-07-30 13:19:20 +01:00
|
|
|
const char *mg_unlist(size_t no) {
|
|
|
|
(void) no;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#endif
|
2021-07-28 21:11:07 +01:00
|
|
|
|
2021-07-30 13:19:20 +01:00
|
|
|
static int is_dir_prefix(const char *prefix, size_t n, const char *path) {
|
2022-02-14 17:44:43 +00:00
|
|
|
// MG_INFO(("[%.*s] [%s] %c", (int) n, prefix, path, path[n]));
|
|
|
|
return n < strlen(path) && strncmp(prefix, path, n) == 0 &&
|
|
|
|
(n == 0 || path[n] == '/' || path[n - 1] == '/');
|
2021-07-30 13:19:20 +01:00
|
|
|
}
|
|
|
|
|
2021-07-29 14:21:20 +01:00
|
|
|
static int packed_stat(const char *path, size_t *size, time_t *mtime) {
|
2021-07-30 13:19:20 +01:00
|
|
|
const char *p;
|
|
|
|
size_t i, n = strlen(path);
|
|
|
|
if (mg_unpack(path, size, mtime)) return MG_FS_READ; // Regular file
|
|
|
|
// Scan all files. If `path` is a dir prefix for any of them, it's a dir
|
|
|
|
for (i = 0; (p = mg_unlist(i)) != NULL; i++) {
|
|
|
|
if (is_dir_prefix(path, n, p)) return MG_FS_DIR;
|
|
|
|
}
|
|
|
|
return 0;
|
2021-07-28 21:11:07 +01:00
|
|
|
}
|
|
|
|
|
2021-07-30 17:41:23 +01:00
|
|
|
static void packed_list(const char *dir, void (*fn)(const char *, void *),
|
2021-07-28 21:11:07 +01:00
|
|
|
void *userdata) {
|
2022-08-27 12:31:43 +01:00
|
|
|
char buf[MG_PATH_MAX], tmp[sizeof(buf)];
|
2021-07-30 17:41:23 +01:00
|
|
|
const char *path, *begin, *end;
|
|
|
|
size_t i, n = strlen(dir);
|
|
|
|
tmp[0] = '\0'; // Previously listed entry
|
|
|
|
for (i = 0; (path = mg_unlist(i)) != NULL; i++) {
|
|
|
|
if (!is_dir_prefix(dir, n, path)) continue;
|
|
|
|
begin = &path[n + 1];
|
|
|
|
end = strchr(begin, '/');
|
|
|
|
if (end == NULL) end = begin + strlen(begin);
|
2022-02-10 11:56:55 +00:00
|
|
|
mg_snprintf(buf, sizeof(buf), "%.*s", (int) (end - begin), begin);
|
2021-07-30 17:41:23 +01:00
|
|
|
buf[sizeof(buf) - 1] = '\0';
|
|
|
|
// If this entry has been already listed, skip
|
|
|
|
// NOTE: we're assuming that file list is sorted alphabetically
|
|
|
|
if (strcmp(buf, tmp) == 0) continue;
|
|
|
|
fn(buf, userdata); // Not yet listed, call user function
|
|
|
|
strcpy(tmp, buf); // And save this entry as listed
|
2021-07-30 13:19:20 +01:00
|
|
|
}
|
2021-07-28 21:11:07 +01:00
|
|
|
}
|
|
|
|
|
2022-01-17 14:42:41 +00:00
|
|
|
static void *packed_open(const char *path, int flags) {
|
2021-07-28 21:11:07 +01:00
|
|
|
size_t size = 0;
|
2021-07-30 13:19:20 +01:00
|
|
|
const char *data = mg_unpack(path, &size, NULL);
|
2021-07-28 21:11:07 +01:00
|
|
|
struct packed_file *fp = NULL;
|
|
|
|
if (data == NULL) return NULL;
|
|
|
|
if (flags & MG_FS_WRITE) return NULL;
|
2023-04-27 11:19:02 -03:00
|
|
|
if ((fp = (struct packed_file *) calloc(1, sizeof(*fp))) != NULL) {
|
|
|
|
fp->size = size;
|
|
|
|
fp->data = data;
|
|
|
|
}
|
2022-01-17 14:42:41 +00:00
|
|
|
return (void *) fp;
|
2021-07-28 21:11:07 +01:00
|
|
|
}
|
|
|
|
|
2022-01-17 14:42:41 +00:00
|
|
|
static void packed_close(void *fp) {
|
|
|
|
if (fp != NULL) free(fp);
|
2021-07-28 21:11:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static size_t packed_read(void *fd, void *buf, size_t len) {
|
|
|
|
struct packed_file *fp = (struct packed_file *) fd;
|
|
|
|
if (fp->pos + len > fp->size) len = fp->size - fp->pos;
|
|
|
|
memcpy(buf, &fp->data[fp->pos], len);
|
|
|
|
fp->pos += len;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t packed_write(void *fd, const void *buf, size_t len) {
|
|
|
|
(void) fd, (void) buf, (void) len;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t packed_seek(void *fd, size_t offset) {
|
|
|
|
struct packed_file *fp = (struct packed_file *) fd;
|
|
|
|
fp->pos = offset;
|
|
|
|
if (fp->pos > fp->size) fp->pos = fp->size;
|
|
|
|
return fp->pos;
|
|
|
|
}
|
|
|
|
|
2022-01-18 17:11:02 +00:00
|
|
|
static bool packed_rename(const char *from, const char *to) {
|
|
|
|
(void) from, (void) to;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool packed_remove(const char *path) {
|
|
|
|
(void) path;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-01-18 19:19:34 +00:00
|
|
|
static bool packed_mkdir(const char *path) {
|
|
|
|
(void) path;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct mg_fs mg_fs_packed = {
|
|
|
|
packed_stat, packed_list, packed_open, packed_close, packed_read,
|
|
|
|
packed_write, packed_seek, packed_rename, packed_remove, packed_mkdir};
|