Add FUSE_CAP_DIRECT_IO_ALLOW_MMAP and use in passthrough_hp
authorBernd Schubert <bschubert@ddn.com>
Mon, 4 Dec 2023 14:58:45 +0000 (15:58 +0100)
committerNikolaus Rath <Nikolaus@rath.org>
Wed, 10 Jan 2024 20:59:03 +0000 (20:59 +0000)
This is not called FUSE_CAP_DIRECT_IO_RELAX, as the kernel flag
FUSE_DIRECT_IO_RELAX is supposed to be renamed to
FUSE_DIRECT_IO_ALLOW_MMAP. The corresponding kernel patches just
did not land yet.

example/passthrough_hp.cc
include/fuse_common.h
lib/fuse_lowlevel.c

index 96201d6bc0c3ddba64d06ea019c2ff88c1059a78..66fe6f827101d35fdfea76410c74d27f5ef46bb0 100644 (file)
@@ -212,6 +212,10 @@ static void sfs_init(void *userdata, fuse_conn_info *conn) {
         if (conn->capable & FUSE_CAP_SPLICE_READ)
             conn->want |= FUSE_CAP_SPLICE_READ;
     }
+
+    /* This is a local file system - no network coherency needed */
+    if (conn->capable & FUSE_CAP_DIRECT_IO_ALLOW_MMAP)
+        conn->want |= FUSE_CAP_DIRECT_IO_ALLOW_MMAP;
 }
 
 
index 3ff14ec90c46b9085caf32f2738e861f5bf9bdfc..a804134c201e60ebbcb752478429386e3f232a92 100644 (file)
@@ -440,6 +440,15 @@ struct fuse_loop_config_v1 {
  */
 #define FUSE_CAP_SETXATTR_EXT     (1 << 27)
 
+/**
+ * Files opened with FUSE_DIRECT_IO do not support MAP_SHARED mmap. This restriction
+ * is relaxed through FUSE_CAP_DIRECT_IO_RELAX (kernel flag: FUSE_DIRECT_IO_RELAX).
+ * MAP_SHARED is disabled by default for FUSE_DIRECT_IO, as this flag can be used to
+ * ensure coherency between mount points (or network clients) and with kernel page
+ * cache as enforced by mmap that cannot be guaranteed anymore.
+ */
+#define FUSE_CAP_DIRECT_IO_ALLOW_MMAP  (1 << 27)
+
 /**
  * Ioctl flags
  *
index ea985360d2ea242d72d84a49db0f52a5b53e4dfa..41e626b4c893b20a8bd180260998a680a74340d7 100644 (file)
@@ -2019,6 +2019,8 @@ void do_init(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)
                                bufsize = max_bufsize;
                        }
                }
+               if (inargflags & FUSE_DIRECT_IO_ALLOW_MMAP)
+                       se->conn.capable |= FUSE_CAP_DIRECT_IO_ALLOW_MMAP;
                if (arg->minor >= 38)
                        se->conn.capable |= FUSE_CAP_EXPIRE_ONLY;
        } else {
@@ -2151,6 +2153,8 @@ void do_init(fuse_req_t req, fuse_ino_t nodeid, const void *inarg)
                outargflags |= FUSE_EXPLICIT_INVAL_DATA;
        if (se->conn.want & FUSE_CAP_SETXATTR_EXT)
                outargflags |= FUSE_SETXATTR_EXT;
+       if (se->conn.want & FUSE_CAP_DIRECT_IO_ALLOW_MMAP)
+               outargflags |= FUSE_DIRECT_IO_ALLOW_MMAP;
 
        if (inargflags & FUSE_INIT_EXT) {
                outargflags |= FUSE_INIT_EXT;