do_init(): treat command line options consistently
authorNikolaus Rath <Nikolaus@rath.org>
Mon, 10 Oct 2016 22:52:15 +0000 (15:52 -0700)
committerNikolaus Rath <Nikolaus@rath.org>
Thu, 13 Oct 2016 17:35:12 +0000 (10:35 -0700)
Previously, some command line options would change the FUSE defaults
but leave the final value to the file systems `init` handler while
others would override any changes made by `init`. Now, command line
options do both: they modify the default, *and* take precedence.

ChangeLog.rst
include/fuse_common.h
include/fuse_lowlevel.h
lib/fuse_lowlevel.c

index dc243f227fa155ad4b7362a9589546938b9a526c..4f355ce3c7af9b24b4eadc6ff631c4cec04a3dcf 100644 (file)
@@ -1,6 +1,12 @@
 Unreleased Changes
 ==================
 
+* `fuse_session_new` now treats low-level options more consistently:
+  First, options are used to modify FUSE defaults. Second, the file
+  system may inspect and/or adjust the settings in its `init`
+  handler. Third, command line arguments take precedence over any
+  modifications made by the `init` handler.
+
 * Removed the `async_read` field from `struct fuse_conn_info`. To
   determine if the kernel supports asynchronous reads, file systems
   should check the `FUSE_CAP_ASYNC_READ` bit of the `capable`
index 4a0e94bcf0effbe9c95ae38cab2e1990b881cfb7..55f0de2d5d1b7d8994085a3736bc93820f1677a8 100644 (file)
@@ -167,12 +167,15 @@ struct fuse_conn_info {
        unsigned max_readahead;
 
        /**
-        * Capability flags, that the kernel supports
+        * Capability flags that the kernel supports (read-only)
         */
        unsigned capable;
 
        /**
-        * Capability flags, that the filesystem wants to enable
+        * Capability flags that the filesystem wants to enable.
+        *
+        * libfuse attempts to initialize this field with
+        * reasonable default values before calling the init() handler.
         */
        unsigned want;
 
index 63c59c9cbce4297fe7c8fce148c93f959b5f4627..bad1d45f9a2f723b57a15c277bdb700de2431c2c 100644 (file)
@@ -165,7 +165,14 @@ struct fuse_lowlevel_ops {
        /**
         * Initialize filesystem
         *
-        * Called before any other filesystem method
+        * This function is called when libfuse establishes
+        * communication with the FUSE kernel module. The file system
+        * should use this module to inspect and/or modify the
+        * connection parameters provided in the `conn` structure.
+        *
+        * Note that some parameters may be overwritten by options
+        * passed to fuse_session_new() which take precedence over the
+        * values set in this handler.
         *
         * There's no reply to this function
         *
index 299d4a87c4b63c1e5c11afda808732fd1cb2bfbb..3b30b21b8f4b131ad066f90de44f6c2a19dd8247 100644 (file)
@@ -1809,6 +1809,39 @@ static void do_fallocate(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)
                fuse_reply_err(req, ENOSYS);
 }
 
+static void apply_want_options(struct fuse_session *opts,
+                       struct fuse_conn_info *conn)
+{
+#define LL_ENABLE(cond,cap) \
+       if (cond) conn->want |= (cap)
+#define LL_DISABLE(cond,cap) \
+       if (cond) conn->want &= ~(cap)
+
+       LL_ENABLE(opts->splice_read, FUSE_CAP_SPLICE_READ);
+       LL_DISABLE(opts->no_splice_read, FUSE_CAP_SPLICE_READ);
+
+       LL_ENABLE(opts->splice_write, FUSE_CAP_SPLICE_WRITE);
+       LL_DISABLE(opts->no_splice_write, FUSE_CAP_SPLICE_WRITE);
+
+       LL_ENABLE(opts->splice_move, FUSE_CAP_SPLICE_MOVE);
+       LL_DISABLE(opts->no_splice_move, FUSE_CAP_SPLICE_MOVE);
+
+       LL_ENABLE(opts->auto_inval_data, FUSE_CAP_AUTO_INVAL_DATA);
+       LL_DISABLE(opts->no_auto_inval_data, FUSE_CAP_AUTO_INVAL_DATA);
+
+       LL_DISABLE(opts->no_readdirplus, FUSE_CAP_READDIRPLUS);
+       LL_DISABLE(opts->no_readdirplus_auto, FUSE_CAP_READDIRPLUS_AUTO);
+
+       LL_ENABLE(opts->async_dio, FUSE_CAP_ASYNC_DIO);
+       LL_DISABLE(opts->no_async_dio, FUSE_CAP_ASYNC_DIO);
+
+       LL_ENABLE(opts->writeback_cache, FUSE_CAP_WRITEBACK_CACHE);
+       LL_DISABLE(opts->no_writeback_cache, FUSE_CAP_WRITEBACK_CACHE);
+
+       LL_ENABLE(opts->async_read, FUSE_CAP_ASYNC_READ);
+       LL_DISABLE(opts->sync_read, FUSE_CAP_ASYNC_READ);
+}
+
 static void do_init(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)
 {
        struct fuse_init_in *arg = (struct fuse_init_in *) inarg;
@@ -1851,11 +1884,8 @@ static void do_init(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)
        if (arg->minor >= 6) {
                if (arg->max_readahead < f->conn.max_readahead)
                        f->conn.max_readahead = arg->max_readahead;
-               if (arg->flags & FUSE_ASYNC_READ) {
+               if (arg->flags & FUSE_ASYNC_READ)
                        f->conn.capable |= FUSE_CAP_ASYNC_READ;
-                       /* Enable by default */
-                       f->conn.want |= FUSE_CAP_ASYNC_READ;
-               }
                if (arg->flags & FUSE_POSIX_LOCKS)
                        f->conn.capable |= FUSE_CAP_POSIX_LOCKS;
                if (arg->flags & FUSE_ATOMIC_O_TRUNC)
@@ -1886,36 +1916,23 @@ static void do_init(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)
 #ifdef HAVE_SPLICE
 #ifdef HAVE_VMSPLICE
                f->conn.capable |= FUSE_CAP_SPLICE_WRITE | FUSE_CAP_SPLICE_MOVE;
-               if (f->splice_write)
-                       f->conn.want |= FUSE_CAP_SPLICE_WRITE;
-               if (f->splice_move)
-                       f->conn.want |= FUSE_CAP_SPLICE_MOVE;
 #endif
                f->conn.capable |= FUSE_CAP_SPLICE_READ;
-               if (f->splice_read)
-                       f->conn.want |= FUSE_CAP_SPLICE_READ;
 #endif
        }
        if (f->conn.proto_minor >= 18)
                f->conn.capable |= FUSE_CAP_IOCTL_DIR;
 
-       if (f->atomic_o_trunc)
-               f->conn.want |= FUSE_CAP_ATOMIC_O_TRUNC;
-       if (f->op.getlk && f->op.setlk && !f->no_remote_posix_lock)
-               f->conn.want |= FUSE_CAP_POSIX_LOCKS;
-       if (f->op.flock && !f->no_remote_flock)
-               f->conn.want |= FUSE_CAP_FLOCK_LOCKS;
-       if (f->auto_inval_data)
-               f->conn.want |= FUSE_CAP_AUTO_INVAL_DATA;
-       if (f->op.readdirplus && !f->no_readdirplus) {
-               f->conn.want |= FUSE_CAP_READDIRPLUS;
-               if (!f->no_readdirplus_auto)
-                       f->conn.want |= FUSE_CAP_READDIRPLUS_AUTO;
-       }
-       if (f->async_dio)
-               f->conn.want |= FUSE_CAP_ASYNC_DIO;
-       if (f->writeback_cache)
-               f->conn.want |= FUSE_CAP_WRITEBACK_CACHE;
+       /* Default settings (where non-zero) */
+#define LL_SET_DEFAULT(cond, cap) \
+       if ((cond) && (f->conn.capable & (cap))) \
+               f->conn.want |= (cap)
+       LL_SET_DEFAULT(1, FUSE_CAP_ASYNC_READ);
+       LL_SET_DEFAULT(f->op.getlk && f->op.setlk,
+                      FUSE_CAP_POSIX_LOCKS);
+       LL_SET_DEFAULT(f->op.flock, FUSE_CAP_FLOCK_LOCKS);
+       LL_SET_DEFAULT(f->op.readdirplus, FUSE_CAP_READDIRPLUS);
+       LL_SET_DEFAULT(f->op.readdirplus, FUSE_CAP_READDIRPLUS_AUTO);
 
        if (bufsize < FUSE_MIN_READ_BUFFER) {
                fprintf(stderr, "fuse: warning: buffer size too small: %zu\n",
@@ -1928,29 +1945,18 @@ static void do_init(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)
                f->conn.max_write = bufsize;
 
        f->got_init = 1;
+
+       /* Apply command-line options (so that init() handler has
+          an idea about user preferences */
+       apply_want_options(f, &f->conn);
+
+       /* Allow file-system to overwrite defaults */
        if (f->op.init)
                f->op.init(f->userdata, &f->conn);
 
-       if (f->no_splice_read)
-               f->conn.want &= ~FUSE_CAP_SPLICE_READ;
-       if (f->no_splice_write)
-               f->conn.want &= ~FUSE_CAP_SPLICE_WRITE;
-       if (f->no_splice_move)
-               f->conn.want &= ~FUSE_CAP_SPLICE_MOVE;
-       if (f->no_auto_inval_data)
-               f->conn.want &= ~FUSE_CAP_AUTO_INVAL_DATA;
-       if (f->no_readdirplus)
-               f->conn.want &= ~FUSE_CAP_READDIRPLUS;
-       if (f->no_readdirplus_auto)
-               f->conn.want &= ~FUSE_CAP_READDIRPLUS_AUTO;
-       if (f->no_async_dio)
-               f->conn.want &= ~FUSE_CAP_ASYNC_DIO;
-       if (f->no_writeback_cache)
-               f->conn.want &= ~FUSE_CAP_WRITEBACK_CACHE;
-       if (f->async_read)
-               f->conn.want |= FUSE_CAP_ASYNC_READ;
-       if (f->sync_read)
-               f->conn.want &= ~FUSE_CAP_ASYNC_READ;
+       /* Now explicitly overwrite file-system's decision
+          with command-line options */
+       apply_want_options(f, &f->conn);
 
        /* Always enable big writes, this is superseded
           by the max_write option */