/* $OpenBSD: fuse.c,v 1.43 2017/12/18 14:20:23 helg Exp $ */ /* * Copyright (c) 2013 Sylvestre Gallon * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include #include #include "fuse_opt.h" #include "fuse_private.h" #include "debug.h" static volatile sig_atomic_t sigraised = 0; static volatile sig_atomic_t signum = 0; static struct fuse_context *ictx = NULL; enum { KEY_DEBUG, KEY_FOREGROUND, KEY_HELP, KEY_HELP_WITHOUT_HEADER, KEY_VERSION, KEY_MAXREAD, KEY_STUB }; /* options supported by fuse_parse_cmdline */ static struct fuse_opt fuse_core_opts[] = { FUSE_OPT_KEY("-d", KEY_DEBUG), FUSE_OPT_KEY("debug", KEY_DEBUG), FUSE_OPT_KEY("-f", KEY_FOREGROUND), FUSE_OPT_KEY("-h", KEY_HELP), FUSE_OPT_KEY("--help", KEY_HELP), FUSE_OPT_KEY("-ho", KEY_HELP_WITHOUT_HEADER), FUSE_OPT_KEY("-s", KEY_STUB), FUSE_OPT_KEY("-V", KEY_VERSION), FUSE_OPT_KEY("--version", KEY_VERSION), FUSE_OPT_END }; /* options supported by fuse_new */ #define FUSE_LIB_OPT(o, m) {o, offsetof(struct fuse_config, m), 1} static struct fuse_opt fuse_lib_opts[] = { FUSE_OPT_KEY("ac_attr_timeout=", KEY_STUB), FUSE_OPT_KEY("allow_other", KEY_STUB), FUSE_OPT_KEY("allow_root", KEY_STUB), FUSE_OPT_KEY("attr_timeout=", KEY_STUB), FUSE_OPT_KEY("auto_cache", KEY_STUB), FUSE_OPT_KEY("noauto_cache", KEY_STUB), FUSE_OPT_KEY("big_writes", KEY_STUB), FUSE_OPT_KEY("debug", KEY_DEBUG), FUSE_OPT_KEY("-d", KEY_DEBUG), FUSE_OPT_KEY("entry_timeout=", KEY_STUB), FUSE_LIB_OPT("gid=", set_gid), FUSE_LIB_OPT("gid=%u", gid), FUSE_OPT_KEY("hard_remove", KEY_STUB), FUSE_OPT_KEY("intr_signal", KEY_STUB), FUSE_OPT_KEY("kernel_cache", KEY_STUB), FUSE_OPT_KEY("large_read", KEY_STUB), FUSE_OPT_KEY("modules=", KEY_STUB), FUSE_OPT_KEY("negative_timeout=", KEY_STUB), FUSE_OPT_KEY("readdir_ino", KEY_STUB), FUSE_OPT_KEY("relatime", KEY_STUB), FUSE_OPT_KEY("subtype=", KEY_STUB), FUSE_LIB_OPT("uid=", set_uid), FUSE_LIB_OPT("uid=%u", uid), FUSE_OPT_KEY("use_ino", KEY_STUB), FUSE_OPT_KEY("dmask=%o", KEY_STUB), FUSE_OPT_KEY("fmask=%o", KEY_STUB), FUSE_LIB_OPT("umask=", set_mode), FUSE_LIB_OPT("umask=%o", umask), FUSE_OPT_END }; /* options supported by fuse_mount */ #define FUSE_MOUNT_OPT(o, m) {o, offsetof(struct fuse_mount_opts, m), 1} static struct fuse_opt fuse_mount_opts[] = { FUSE_OPT_KEY("async_read", KEY_STUB), FUSE_OPT_KEY("blkdev", KEY_STUB), FUSE_OPT_KEY("blksize=", KEY_STUB), FUSE_MOUNT_OPT("default_permissions", def_perms), FUSE_OPT_KEY("direct_io", KEY_STUB), FUSE_MOUNT_OPT("fsname=%s", fsname), FUSE_MOUNT_OPT("max_read=%u", max_read), FUSE_OPT_KEY("max_readahead", KEY_STUB), FUSE_OPT_KEY("max_write", KEY_STUB), FUSE_MOUNT_OPT("noatime", noatime), FUSE_MOUNT_OPT("nonempty", nonempty), FUSE_MOUNT_OPT("-r", rdonly), FUSE_MOUNT_OPT("ro", rdonly), FUSE_OPT_KEY("ro_fallback", KEY_STUB), FUSE_OPT_KEY("sync_read", KEY_STUB), FUSE_OPT_END }; static void ifuse_sighdlr(int num) { if (!sigraised || (num == SIGCHLD)) { sigraised = 1; signum = num; } } static void ifuse_try_unmount(struct fuse *f) { pid_t child; signal(SIGCHLD, ifuse_sighdlr); /* unmount in another thread so fuse_loop() doesn't deadlock */ child = fork(); if (child < 0) { DPERROR(__func__); return; } if (child == 0) { fuse_remove_signal_handlers(fuse_get_session(f)); errno = 0; fuse_unmount(f->fc->dir, f->fc); _exit(errno); } } static void ifuse_child_exit(const struct fuse *f) { int status; signal(SIGCHLD, SIG_DFL); if (waitpid(WAIT_ANY, &status, WNOHANG) == -1) fprintf(stderr, "fuse: %s\n", strerror(errno)); if (WIFEXITED(status) && (WEXITSTATUS(status) != 0)) fprintf(stderr, "fuse: %s: %s\n", f->fc->dir, strerror(WEXITSTATUS(status))); sigraised = 0; return; } int fuse_loop(struct fuse *fuse) { struct fusebuf fbuf; struct fuse_context ctx; struct fb_ioctl_xch ioexch; struct kevent ev; ssize_t n; int ret; if (fuse == NULL) return (-1); fuse->fc->kq = kqueue(); if (fuse->fc->kq == -1) return (-1); EV_SET(&fuse->fc->event, fuse->fc->fd, EVFILT_READ, EV_ADD | EV_ENABLE, 0, 0, 0); while (!fuse->fc->dead) { ret = kevent(fuse->fc->kq, &fuse->fc->event, 1, &ev, 1, NULL); if (ret == -1) { if (errno == EINTR) { switch (signum) { case SIGCHLD: ifuse_child_exit(fuse); break; case SIGHUP: case SIGINT: case SIGTERM: ifuse_try_unmount(fuse); break; default: fprintf(stderr, "%s: %s\n", __func__, strsignal(signum)); } } else DPERROR(__func__); } else if (ret > 0) { n = read(fuse->fc->fd, &fbuf, sizeof(fbuf)); if (n != sizeof(fbuf)) { fprintf(stderr, "%s: bad fusebuf read\n", __func__); return (-1); } /* check if there is data something present */ if (fbuf.fb_len) { fbuf.fb_dat = malloc(fbuf.fb_len); if (fbuf.fb_dat == NULL) return (-1); ioexch.fbxch_uuid = fbuf.fb_uuid; ioexch.fbxch_len = fbuf.fb_len; ioexch.fbxch_data = fbuf.fb_dat; if (ioctl(fuse->fc->fd, FIOCGETFBDAT, &ioexch)) { free(fbuf.fb_dat); return (-1); } } ctx.fuse = fuse; ctx.uid = fuse->conf.uid; ctx.gid = fuse->conf.gid; ctx.pid = fuse->conf.pid; ctx.umask = fuse->conf.umask; ctx.private_data = fuse->private_data; ictx = &ctx; ret = ifuse_exec_opcode(fuse, &fbuf); if (ret) { ictx = NULL; return (-1); } n = write(fuse->fc->fd, &fbuf, sizeof(fbuf)); if (fbuf.fb_len) { if (fbuf.fb_dat == NULL) { fprintf(stderr, "%s: fb_dat is Null\n", __func__); return (-1); } ioexch.fbxch_uuid = fbuf.fb_uuid; ioexch.fbxch_len = fbuf.fb_len; ioexch.fbxch_data = fbuf.fb_dat; if (ioctl(fuse->fc->fd, FIOCSETFBDAT, &ioexch)) { free(fbuf.fb_dat); return (-1); } free(fbuf.fb_dat); } ictx = NULL; if (n != FUSEBUFSIZE) { errno = EINVAL; return (-1); } } } return (0); } DEF(fuse_loop); struct fuse_chan * fuse_mount(const char *dir, struct fuse_args *args) { struct fusefs_args fargs; struct fuse_mount_opts opts; struct fuse_chan *fc; const char *errcause; int mnt_flags; if (dir == NULL) return (NULL); fc = calloc(1, sizeof(*fc)); if (fc == NULL) return (NULL); fc->dir = realpath(dir, NULL); if (fc->dir == NULL) goto bad; if ((fc->fd = open("/dev/fuse0", O_RDWR)) == -1) { perror(__func__); goto bad; } bzero(&opts, sizeof(opts)); if (fuse_opt_parse(args, &opts, fuse_mount_opts, NULL) == -1) goto bad; mnt_flags = 0; if (opts.rdonly) mnt_flags |= MNT_RDONLY; if (opts.noatime) mnt_flags |= MNT_NOATIME; if (opts.max_read > FUSEBUFMAXSIZE) { fprintf(stderr, "fuse: invalid max_read (%d > %d)\n", opts.max_read, FUSEBUFMAXSIZE); goto bad; } bzero(&fargs, sizeof(fargs)); fargs.fd = fc->fd; fargs.max_read = opts.max_read; if (mount(MOUNT_FUSEFS, fc->dir, mnt_flags, &fargs)) { switch (errno) { case EMFILE: errcause = "mount table full"; break; case EOPNOTSUPP: errcause = "filesystem not supported by kernel"; break; default: errcause = strerror(errno); break; } fprintf(stderr, "%s on %s: %s\n", __func__, dir, errcause); goto bad; } return (fc); bad: if (fc->fd != -1) close(fc->fd); free(fc->dir); free(fc); return (NULL); } DEF(fuse_mount); void fuse_unmount(const char *dir, struct fuse_chan *ch) { if (ch == NULL || ch->dead) return; if (unmount(dir, MNT_UPDATE) == -1) DPERROR(__func__); } DEF(fuse_unmount); int fuse_is_lib_option(const char *opt) { return (fuse_opt_match(fuse_lib_opts, opt)); } int fuse_chan_fd(struct fuse_chan *ch) { if (ch == NULL) return (-1); return (ch->fd); } struct fuse_session * fuse_get_session(struct fuse *f) { return (&f->se); } DEF(fuse_get_session); int fuse_loop_mt(unused struct fuse *fuse) { return (-1); } static int ifuse_lib_opt_proc(void *data, const char *arg, int key, unused struct fuse_args *args) { switch (key) { case KEY_STUB: return (0); case KEY_DEBUG: ifuse_debug_init(); break; default: fprintf(stderr, "fuse: unrecognised option %s\n", arg); return (-1); } /* Keep unknown options. */ return (1); } struct fuse * fuse_new(struct fuse_chan *fc, struct fuse_args *args, const struct fuse_operations *ops, unused size_t size, void *userdata) { struct fuse *fuse; struct fuse_vnode *root; if (fc == NULL || ops == NULL) return (NULL); if ((fuse = calloc(1, sizeof(*fuse))) == NULL) return (NULL); /* copy fuse ops to their own structure */ memcpy(&fuse->op, ops, sizeof(fuse->op)); if (fuse_opt_parse(args, &fuse->conf, fuse_lib_opts, ifuse_lib_opt_proc) == -1) { free(fuse); return (NULL); } fuse->fc = fc; fuse->max_ino = FUSE_ROOT_INO; fuse->se.args = fuse; fuse->private_data = userdata; if ((root = alloc_vn(fuse, "/", FUSE_ROOT_INO, 0)) == NULL) { free(fuse); return (NULL); } tree_init(&fuse->vnode_tree); tree_init(&fuse->name_tree); if (!set_vn(fuse, root)) { free(fuse); return (NULL); } return (fuse); } DEF(fuse_new); int fuse_daemonize(int foreground) { if (foreground) return (0); return (daemon(0, 0)); } DEF(fuse_daemonize); void fuse_destroy(struct fuse *f) { if (f == NULL) return; /* * Even though these were allocated in fuse_mount(), we can't free them * in fuse_unmount() since fuse_loop() will not have terminated yet so * we free them here. */ close(f->fc->fd); free(f->fc->dir); free(f->fc); free(f); } DEF(fuse_destroy); void fuse_remove_signal_handlers(unused struct fuse_session *se) { signal(SIGHUP, SIG_DFL); signal(SIGINT, SIG_DFL); signal(SIGTERM, SIG_DFL); signal(SIGPIPE, SIG_DFL); } DEF(fuse_remove_signal_handlers); int fuse_set_signal_handlers(unused struct fuse_session *se) { signal(SIGHUP, ifuse_sighdlr); signal(SIGINT, ifuse_sighdlr); signal(SIGTERM, ifuse_sighdlr); signal(SIGPIPE, SIG_IGN); return (0); } static void dump_help(void) { fprintf(stderr, "FUSE options:\n" " -d -o debug enable debug output (implies -f)\n" " -f run in foreground\n" " -V --version print fuse version\n" "\n"); } static void dump_version(void) { fprintf(stderr, "FUSE library version: %d.%d\n", FUSE_MAJOR_VERSION, FUSE_MINOR_VERSION); } static int ifuse_process_opt(void *data, const char *arg, int key, unused struct fuse_args *args) { struct fuse_core_opts *opt = data; struct stat st; int res; switch (key) { case KEY_STUB: return (0); case KEY_DEBUG: ifuse_debug_init(); /* falls through */ case KEY_FOREGROUND: opt->foreground = 1; return (0); case KEY_HELP: case KEY_HELP_WITHOUT_HEADER: dump_help(); return (-1); case KEY_VERSION: dump_version(); return (-1); case FUSE_OPT_KEY_NONOPT: if (opt->mp == NULL) { opt->mp = realpath(arg, opt->mp); if (opt->mp == NULL) { fprintf(stderr, "fuse: realpath: " "%s : %s\n", arg, strerror(errno)); return (-1); } res = stat(opt->mp, &st); if (res == -1) { fprintf(stderr, "fuse: bad mount point " "%s : %s\n", arg, strerror(errno)); return (-1); } if (!S_ISDIR(st.st_mode)) { fprintf(stderr, "fuse: bad mount point " "%s : %s\n", arg, strerror(ENOTDIR)); return (-1); } } return (0); } /* Pass through unknown options. */ return (1); } int fuse_parse_cmdline(struct fuse_args *args, char **mp, int *mt, int *fg) { struct fuse_core_opts opt; bzero(&opt, sizeof(opt)); if (fuse_opt_parse(args, &opt, fuse_core_opts, ifuse_process_opt) == -1) return (-1); if (opt.mp == NULL) { fprintf(stderr, "fuse: missing mountpoint parameter\n"); return (-1); } if (mp != NULL) { *mp = strdup(opt.mp); if (*mp == NULL) return (-1); } if (mt != NULL) *mt = 0; if (fg != NULL) *fg = opt.foreground; return (0); } DEF(fuse_parse_cmdline); struct fuse_context * fuse_get_context(void) { return (ictx); } DEF(fuse_get_context); int fuse_version(void) { return (FUSE_VERSION); } void fuse_teardown(struct fuse *fuse, char *mp) { if (fuse == NULL || mp == NULL) return; fuse_remove_signal_handlers(fuse_get_session(fuse)); fuse_unmount(mp, fuse->fc); fuse_destroy(fuse); } int fuse_invalidate(unused struct fuse *f, unused const char *path) { return (EINVAL); } struct fuse * fuse_setup(int argc, char **argv, const struct fuse_operations *ops, size_t size, char **mp, int *mt, void *data) { struct fuse_args args = FUSE_ARGS_INIT(argc, argv); struct fuse_chan *fc; struct fuse *fuse; char *dir; int fg; dir = NULL; if (fuse_parse_cmdline(&args, &dir, mt, &fg)) goto err; fuse_daemonize(fg); if ((fc = fuse_mount(dir, &args)) == NULL) goto err; if ((fuse = fuse_new(fc, &args, ops, size, data)) == NULL) { fuse_unmount(dir, fc); close(fc->fd); free(fc->dir); free(fc); goto err; } /* args are no longer needed */ fuse_opt_free_args(&args); if (fuse_set_signal_handlers(fuse_get_session(fuse)) == -1) { fuse_unmount(dir, fc); fuse_destroy(fuse); goto err; } /* the caller frees dir, but we do it if the caller doesn't want it */ if (mp == NULL) free(dir); else *mp = dir; return (fuse); err: free(dir); return (NULL); } DEF(fuse_setup); int fuse_main(int argc, char **argv, const struct fuse_operations *ops, void *data) { struct fuse *fuse; fuse = fuse_setup(argc, argv, ops, sizeof(*ops), NULL, NULL, data); if (fuse == NULL) return (-1); return (fuse_loop(fuse)); }