| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * fusectl test file-system
- * Creates a simple FUSE filesystem with a single read-write file (/test)
- */
- #define FUSE_USE_VERSION 26
- #include <fuse.h>
- #include <stdio.h>
- #include <string.h>
- #include <errno.h>
- #include <fcntl.h>
- #include <stdlib.h>
- #include <unistd.h>
- #define MAX(a, b) ((a) > (b) ? (a) : (b))
- static char *content;
- static size_t content_size = 0;
- static const char test_path[] = "/test";
- static int test_getattr(const char *path, struct stat *st)
- {
- memset(st, 0, sizeof(*st));
- if (!strcmp(path, "/")) {
- st->st_mode = S_IFDIR | 0755;
- st->st_nlink = 2;
- return 0;
- }
- if (!strcmp(path, test_path)) {
- st->st_mode = S_IFREG | 0664;
- st->st_nlink = 1;
- st->st_size = content_size;
- return 0;
- }
- return -ENOENT;
- }
- static int test_readdir(const char *path, void *buf, fuse_fill_dir_t filler,
- off_t offset, struct fuse_file_info *fi)
- {
- if (strcmp(path, "/"))
- return -ENOENT;
- filler(buf, ".", NULL, 0);
- filler(buf, "..", NULL, 0);
- filler(buf, test_path + 1, NULL, 0);
- return 0;
- }
- static int test_open(const char *path, struct fuse_file_info *fi)
- {
- if (strcmp(path, test_path))
- return -ENOENT;
- return 0;
- }
- static int test_read(const char *path, char *buf, size_t size, off_t offset,
- struct fuse_file_info *fi)
- {
- if (strcmp(path, test_path) != 0)
- return -ENOENT;
- if (!content || content_size == 0)
- return 0;
- if (offset >= content_size)
- return 0;
- if (offset + size > content_size)
- size = content_size - offset;
- memcpy(buf, content + offset, size);
- return size;
- }
- static int test_write(const char *path, const char *buf, size_t size,
- off_t offset, struct fuse_file_info *fi)
- {
- size_t new_size;
- if (strcmp(path, test_path) != 0)
- return -ENOENT;
- if(offset > content_size)
- return -EINVAL;
- new_size = MAX(offset + size, content_size);
- if (new_size > content_size)
- content = realloc(content, new_size);
- content_size = new_size;
- if (!content)
- return -ENOMEM;
- memcpy(content + offset, buf, size);
- return size;
- }
- static int test_truncate(const char *path, off_t size)
- {
- if (strcmp(path, test_path) != 0)
- return -ENOENT;
- if (size == 0) {
- free(content);
- content = NULL;
- content_size = 0;
- return 0;
- }
- content = realloc(content, size);
- if (!content)
- return -ENOMEM;
- if (size > content_size)
- memset(content + content_size, 0, size - content_size);
- content_size = size;
- return 0;
- }
- static struct fuse_operations memfd_ops = {
- .getattr = test_getattr,
- .readdir = test_readdir,
- .open = test_open,
- .read = test_read,
- .write = test_write,
- .truncate = test_truncate,
- };
- int main(int argc, char *argv[])
- {
- return fuse_main(argc, argv, &memfd_ops, NULL);
- }
|