API update for fuse_loop_config additions
authorBernd Schubert <bschubert@ddn.com>
Fri, 8 Apr 2022 20:30:27 +0000 (22:30 +0200)
committerNikolaus Rath <Nikolaus@rath.org>
Sun, 4 Sep 2022 12:07:15 +0000 (13:07 +0100)
struct fuse_loop_config was passed as a plain struct, without any
version identifer. This had two implications

1) Any addition of new parameters required a FUSE_SYMVER for
fuse_session_loop_mt() and fuse_loop_mt() as otherwise a read
beyond end-of previous struct size might have happened.

2) Filesystems also might have been recompiled and the developer
might not have noticed the struct extensions and unexpected for
the developer (or people recomliling the code) uninitialized
parameters would have been passed.

Code is updated to have struct fuse_loop_config as an opaque/private
data type for file systems that want version 312
(FUSE_MAKE_VERSION(3, 12)). The deprecated fuse_loop_config_v1
is visible, but should not be used outside of internal
conversion functions

File systems that want version >= 32 < 312 get the previous
struct (through ifdefs) and the #define of fuse_loop_mt
and fuse_session_loop_mt ensures that these recompiled file
systems call into the previous API, which then converts
the struct. This is similar to existing compiled applications
when just libfuse updated, but binaries it is solved with
the FUSE_SYMVER ABI compact declarations.

Signed-off-by: Bernd Schubert <bschubert@ddn.com>
13 files changed:
include/fuse.h
include/fuse_common.h
include/fuse_lowlevel.h
lib/cuse_lowlevel.c
lib/fuse.c
lib/fuse_i.h
lib/fuse_loop_mt.c
lib/fuse_misc.h
lib/fuse_versionscript
lib/helper.c
lib/meson.build
meson.build
util/meson.build

index 1a2f841301d6f3cd5c595abef14670df14d020ed..917a91c1c48b1d7669497c0914a2431da05e019c 100644 (file)
@@ -1011,6 +1011,9 @@ void fuse_exit(struct fuse *f);
 #if FUSE_USE_VERSION < 32
 int fuse_loop_mt_31(struct fuse *f, int clone_fd);
 #define fuse_loop_mt(f, clone_fd) fuse_loop_mt_31(f, clone_fd)
+#elif FUSE_USE_VERSION < FUSE_MAKE_VERSION(3, 12)
+int fuse_loop_mt_32(struct fuse *f, struct fuse_loop_config *config);
+#define fuse_loop_mt(f, config) fuse_loop_mt_32(f, config)
 #else
 /**
  * FUSE event loop with multiple threads
index 3f7260cb3ff6229dab624c85e2acc580ffb6bc91..b40814fdf16e57207b3d0ce3706e15ae74f7b354 100644 (file)
@@ -23,7 +23,7 @@
 #define FUSE_MAJOR_VERSION 3
 
 /** Minor version of FUSE library interface */
-#define FUSE_MINOR_VERSION 11
+#define FUSE_MINOR_VERSION 12
 
 #define FUSE_MAKE_VERSION(maj, min)  ((maj) * 100 + (min))
 #define FUSE_VERSION FUSE_MAKE_VERSION(FUSE_MAJOR_VERSION, FUSE_MINOR_VERSION)
@@ -104,11 +104,20 @@ struct fuse_file_info {
        uint32_t poll_events;
 };
 
+
+
 /**
  * Configuration parameters passed to fuse_session_loop_mt() and
  * fuse_loop_mt().
+ * Deprecated and replaced by a newer private struct in FUSE API
+ * version 312 (FUSE_MAKE_VERSION(3, 12)
  */
+#if FUSE_USE_VERSION < FUSE_MAKE_VERSION(3, 12)
+struct fuse_loop_config_v1; /* forward declarition */
 struct fuse_loop_config {
+#else
+struct fuse_loop_config_v1 {
+#endif
        /**
         * whether to use separate device fds for each thread
         * (may increase performance)
@@ -128,6 +137,7 @@ struct fuse_loop_config {
        unsigned int max_idle_threads;
 };
 
+
 /**************************************************************************
  * Capability bits for 'fuse_conn_info.capable' and 'fuse_conn_info.want' *
  **************************************************************************/
@@ -834,6 +844,39 @@ int fuse_set_signal_handlers(struct fuse_session *se);
  */
 void fuse_remove_signal_handlers(struct fuse_session *se);
 
+/**
+ * Create and set default config for fuse_session_loop_mt and fuse_loop_mt.
+ *
+ * @return anonymous config struct
+ */
+struct fuse_loop_config *fuse_loop_cfg_create(void);
+
+/**
+ * Free the config data structure
+ */
+void fuse_loop_cfg_destroy(struct fuse_loop_config *config);
+
+/**
+ * fuse_loop_config2 setter to set the number of max idle threads.
+ */
+void fuse_loop_cfg_set_idle_threads(struct fuse_loop_config *config,
+                                   unsigned int value);
+
+/**
+ * fuse_loop_config2 setter to enable the clone_fd feature
+ */
+void fuse_loop_cfg_set_clone_fd(struct fuse_loop_config *config,
+                               unsigned int value);
+
+/**
+ * Convert old config to more recernt fuse_loop_config2
+ *
+ * @param config current config2 type
+ * @param v1_conf older config1 type (below FUSE API 312)
+ */
+void fuse_loop_cfg_convert(struct fuse_loop_config *config,
+                          struct fuse_loop_config_v1 *v1_conf);
+
 /* ----------------------------------------------------------- *
  * Compatibility stuff                                        *
  * ----------------------------------------------------------- */
index d73e9facc844c1718fefa4fe14c0274d45f309b2..378742dfeadde71770b431988d4638b905c59090 100644 (file)
@@ -1968,26 +1968,29 @@ int fuse_session_mount(struct fuse_session *se, const char *mountpoint);
 int fuse_session_loop(struct fuse_session *se);
 
 #if FUSE_USE_VERSION < 32
-int fuse_session_loop_mt_31(struct fuse_session *se, int clone_fd);
-#define fuse_session_loop_mt(se, clone_fd) fuse_session_loop_mt_31(se, clone_fd)
+       int fuse_session_loop_mt_31(struct fuse_session *se, int clone_fd);
+       #define fuse_session_loop_mt(se, clone_fd) fuse_session_loop_mt_31(se, clone_fd)
+#elif FUSE_USE_VERSION < FUSE_MAKE_VERSION(3, 12)
+       int fuse_session_loop_mt_32(struct fuse_session *se, struct fuse_loop_config *config);
+       #define fuse_session_loop_mt(se, config) fuse_session_loop_mt_32(se, config)
 #else
-#if (!defined(__UCLIBC__) && !defined(__APPLE__))
-/**
- * Enter a multi-threaded event loop.
- *
- * For a description of the return value and the conditions when the
- * event loop exits, refer to the documentation of
- * fuse_session_loop().
- *
- * @param se the session
- * @param config session loop configuration 
- * @return see fuse_session_loop()
- */
-int fuse_session_loop_mt(struct fuse_session *se, struct fuse_loop_config *config);
-#else
-int fuse_session_loop_mt_32(struct fuse_session *se, struct fuse_loop_config *config);
-#define fuse_session_loop_mt(se, config) fuse_session_loop_mt_32(se, config)
-#endif
+       #if (!defined(__UCLIBC__) && !defined(__APPLE__))
+               /**
               * Enter a multi-threaded event loop.
               *
               * For a description of the return value and the conditions when the
               * event loop exits, refer to the documentation of
               * fuse_session_loop().
               *
               * @param se the session
+                * @param config session loop configuration
               * @return see fuse_session_loop()
               */
+               int fuse_session_loop_mt(struct fuse_session *se, struct fuse_loop_config *config);
+       #else
+               int fuse_session_loop_mt_312(struct fuse_session *se, struct fuse_loop_config *config);
+               #define fuse_session_loop_mt(se, config) fuse_session_loop_mt_312(se, config)
+       #endif
 #endif
 
 /**
index b70947e3d1b2135c603462e8eb3bb17e38f8da6e..01a62ab0a28def53c1c6a47ff548902ca93adf7e 100644 (file)
@@ -351,10 +351,9 @@ int cuse_lowlevel_main(int argc, char *argv[], const struct cuse_info *ci,
                return 1;
 
        if (multithreaded) {
-               struct fuse_loop_config config;
-               config.clone_fd = 0;
-               config.max_idle_threads = 10;
-               res = fuse_session_loop_mt_32(se, &config);
+               struct fuse_loop_config *config = fuse_loop_cfg_create();
+               res = fuse_session_loop_mt(se, config);
+               fuse_loop_cfg_destroy(config);
        }
        else
                res = fuse_session_loop(se);
index 1924caa8677e55ef67cef5263ec404b1de539972..17850e314bbd005827386a47e51443e6e5e2e684 100644 (file)
@@ -4577,8 +4577,8 @@ int fuse_loop(struct fuse *f)
        return fuse_session_loop(f->se);
 }
 
-FUSE_SYMVER("fuse_loop_mt_32", "fuse_loop_mt@@FUSE_3.2")
-int fuse_loop_mt_32(struct fuse *f, struct fuse_loop_config *config)
+FUSE_SYMVER("fuse_loop_mt_312", "fuse_loop_mt@@FUSE_3.12")
+int fuse_loop_mt_312(struct fuse *f, struct fuse_loop_config *config)
 {
        if (f == NULL)
                return -1;
@@ -4587,19 +4587,45 @@ int fuse_loop_mt_32(struct fuse *f, struct fuse_loop_config *config)
        if (res)
                return -1;
 
-       res = fuse_session_loop_mt_32(fuse_get_session(f), config);
+       res = fuse_session_loop_mt_312(fuse_get_session(f), config);
        fuse_stop_cleanup_thread(f);
        return res;
 }
 
+int fuse_loop_mt_32(struct fuse *f, struct fuse_loop_config_v1 *config_v1);
+FUSE_SYMVER("fuse_loop_mt_32", "fuse_loop_mt@FUSE_3.2")
+int fuse_loop_mt_32(struct fuse *f, struct fuse_loop_config_v1 *config_v1)
+{
+       struct fuse_loop_config *config = fuse_loop_cfg_create();
+       if (config == NULL)
+               return ENOMEM;
+
+       fuse_loop_cfg_convert(config, config_v1);
+
+       int res = fuse_loop_mt_312(f, config);
+
+       fuse_loop_cfg_destroy(config);
+
+       return res;
+}
+
 int fuse_loop_mt_31(struct fuse *f, int clone_fd);
 FUSE_SYMVER("fuse_loop_mt_31", "fuse_loop_mt@FUSE_3.0")
 int fuse_loop_mt_31(struct fuse *f, int clone_fd)
 {
-       struct fuse_loop_config config;
-       config.clone_fd = clone_fd;
-       config.max_idle_threads = 10;
-       return fuse_loop_mt_32(f, &config);
+       int err;
+       struct fuse_loop_config *config = fuse_loop_cfg_create();
+
+       if (config == NULL)
+               return ENOMEM;
+
+       fuse_loop_cfg_set_clone_fd(config, clone_fd);
+
+       err = fuse_loop_mt_312(f, config);
+
+       fuse_loop_cfg_destroy(config);
+
+       return err;
 }
 
 void fuse_exit(struct fuse *f)
index d38b630ac5fc94c52cc69db469e5349e7311a499..6930a204206f06e8fea94310e8e665ce2f58ceff 100644 (file)
@@ -87,6 +87,50 @@ struct fuse_module {
        int ctr;
 };
 
+/**
+ * Configuration parameters passed to fuse_session_loop_mt() and
+ * fuse_loop_mt().
+ *
+ * Internal API to avoid exposing the plain data structure and
+ * causing compat issues after adding or removing struct members.
+ *
+ */
+#if FUSE_USE_VERSION >= FUSE_MAKE_VERSION(3, 12)
+struct fuse_loop_config
+{
+       /* verififier that a correct struct was was passed. This is especially
+        * needed, as versions below (3, 12) were using a public struct
+        * (now called  fuse_loop_config_v1), which was hard to extend with
+        * additional parameters, without risking that file system implementations
+        * would not have noticed and might either pass uninitialized members
+        * or even too small structs.
+        * fuse_loop_config_v1 has clone_fd at this offset, which should be either 0
+        * or 1. v2 or even higher version just need to set a value here
+        * which not conflicting and very unlikely as having been set by
+        * file system implementation.
+        */
+       int version_id;
+
+       /**
+        * whether to use separate device fds for each thread
+        * (may increase performance)
+        */
+       int clone_fd;
+       /**
+        * The maximum number of available worker threads before they
+        * start to get deleted when they become idle. If not
+        * specified, the default is 10.
+        *
+        * Adjusting this has performance implications; a very small number
+        * of threads in the pool will cause a lot of thread creation and
+        * deletion overhead and performance may suffer. When set to 0, a new
+        * thread will be created to service every operation.
+        * The special value of -1 means that this parameter is disabled.
+        */
+       int max_idle_threads;
+};
+#endif
+
 /* ----------------------------------------------------------- *
  * Channel interface (when using -o clone_fd)                 *
  * ----------------------------------------------------------- */
@@ -128,8 +172,16 @@ void fuse_session_process_buf_int(struct fuse_session *se,
 
 struct fuse *fuse_new_31(struct fuse_args *args, const struct fuse_operations *op,
                      size_t op_size, void *private_data);
-int fuse_loop_mt_32(struct fuse *f, struct fuse_loop_config *config);
-int fuse_session_loop_mt_32(struct fuse_session *se, struct fuse_loop_config *config);
+int fuse_loop_mt_312(struct fuse *f, struct fuse_loop_config *config);
+int fuse_session_loop_mt_312(struct fuse_session *se, struct fuse_loop_config *config);
+
+/**
+ * Internal verifier for the given config.
+ *
+ * @return negative standard error code or 0 on success
+ */
+int fuse_loop_cfg_verify(struct fuse_loop_config *config);
+
 
 #define FUSE_MAX_MAX_PAGES 256
 #define FUSE_DEFAULT_MAX_PAGES_PER_REQ 32
index 8fcc46cc7195a935c981683f753ae1bb5d07ac03..2f0470b3ec8fbe3614a424f92e74077c869d7e2b 100644 (file)
 #include <sys/time.h>
 #include <sys/ioctl.h>
 #include <assert.h>
+#include <limits.h>
 
 /* Environment var controlling the thread stack size */
 #define ENVNAME_THREAD_STACK "FUSE_THREAD_STACK"
 
+#define FUSE_LOOP_MT_V2_IDENTIFIER      INT_MAX - 2
+#define FUSE_LOOP_MT_DEF_CLONE_FD       0
+#define FUSE_LOOP_MT_DEF_IDLE_THREADS -1 /* thread destruction is disabled
+                                          * by default */
+
 struct fuse_worker {
        struct fuse_worker *prev;
        struct fuse_worker *next;
@@ -303,13 +309,18 @@ static void fuse_join_worker(struct fuse_mt *mt, struct fuse_worker *w)
        free(w);
 }
 
-FUSE_SYMVER("fuse_session_loop_mt_32", "fuse_session_loop_mt@@FUSE_3.2")
-int fuse_session_loop_mt_32(struct fuse_session *se, struct fuse_loop_config *config)
+int fuse_session_loop_mt_312(struct fuse_session *se, struct fuse_loop_config *config);
+FUSE_SYMVER("fuse_session_loop_mt_312", "fuse_session_loop_mt@@FUSE_3.12")
+int fuse_session_loop_mt_312(struct fuse_session *se, struct fuse_loop_config *config)
 {
-       int err;
+int err;
        struct fuse_mt mt;
        struct fuse_worker *w;
 
+       err = fuse_loop_cfg_verify(config);
+       if (err)
+               return err;
+
        memset(&mt, 0, sizeof(struct fuse_mt));
        mt.se = se;
        mt.clone_fd = config->clone_fd;
@@ -347,15 +358,83 @@ int fuse_session_loop_mt_32(struct fuse_session *se, struct fuse_loop_config *co
        if(se->error != 0)
                err = se->error;
        fuse_session_reset(se);
+
+       return err;
+}
+
+int fuse_session_loop_mt_32(struct fuse_session *se, struct fuse_loop_config_v1 *config_v1);
+FUSE_SYMVER("fuse_session_loop_mt_32", "fuse_session_loop_mt@FUSE_3.2")
+int fuse_session_loop_mt_32(struct fuse_session *se, struct fuse_loop_config_v1 *config_v1)
+{
+       int err;
+
+       struct fuse_loop_config *config = fuse_loop_cfg_create();
+       if (config == NULL)
+               return ENOMEM;
+
+       fuse_loop_cfg_convert(config, config_v1);
+
+       err = fuse_session_loop_mt_312(se, config);
+
+       fuse_loop_cfg_destroy(config);
+
        return err;
 }
 
+
 int fuse_session_loop_mt_31(struct fuse_session *se, int clone_fd);
 FUSE_SYMVER("fuse_session_loop_mt_31", "fuse_session_loop_mt@FUSE_3.0")
 int fuse_session_loop_mt_31(struct fuse_session *se, int clone_fd)
 {
-       struct fuse_loop_config config;
-       config.clone_fd = clone_fd;
-       config.max_idle_threads = 10;
-       return fuse_session_loop_mt_32(se, &config);
+       struct fuse_loop_config *config = fuse_loop_cfg_create();
+       if (clone_fd > 0)
+                fuse_loop_cfg_set_clone_fd(config, clone_fd);
+       return fuse_session_loop_mt_312(se, config);
+}
+
+struct fuse_loop_config *fuse_loop_cfg_create(void)
+{
+       struct fuse_loop_config *config = calloc(1, sizeof(*config));
+       if (config == NULL)
+               return NULL;
+
+       config->version_id       = FUSE_LOOP_MT_V2_IDENTIFIER;
+       config->max_idle_threads = FUSE_LOOP_MT_DEF_IDLE_THREADS;
+       config->clone_fd         = FUSE_LOOP_MT_DEF_CLONE_FD;
+
+       return config;
+}
+
+void fuse_loop_cfg_destroy(struct fuse_loop_config *config)
+{
+       free(config);
+}
+
+int fuse_loop_cfg_verify(struct fuse_loop_config *config)
+{
+       if (config->version_id != FUSE_LOOP_MT_V2_IDENTIFIER)
+               return -EINVAL;
+
+       return 0;
+}
+
+void fuse_loop_cfg_convert(struct fuse_loop_config *config,
+                          struct fuse_loop_config_v1 *v1_conf)
+{
+       fuse_loop_cfg_set_idle_threads(config, v1_conf->max_idle_threads);
+
+       fuse_loop_cfg_set_clone_fd(config, v1_conf->clone_fd);
+}
+
+void fuse_loop_cfg_set_idle_threads(struct fuse_loop_config *config,
+                                   unsigned int value)
+{
+       config->max_idle_threads = value;
 }
+
+void fuse_loop_cfg_set_clone_fd(struct fuse_loop_config *config,
+                               unsigned int value)
+{
+       config->clone_fd = value;
+}
+
index f956ab7994af2a90ee28326596ea7dcad3ec1131..e2e9ba59c6257c16617ed59e9dfe9f1c51adfeda 100644 (file)
@@ -11,6 +11,9 @@
 /*
   Versioned symbols cannot be used in some cases because it
     - not supported on MacOSX (in MachO binary format)
+
+  Note: "@@" denotes the default symbol, "@" is binary a compat version.
+
 */
 #ifndef __APPLE__
 # if HAVE_SYMVER_ATTRIBUTE
index a06f7689a123b889c7d8d5428f3ba8a103a226eb..aff7e84f696c6c32d75f68d9f0c6721ba509d67f 100644 (file)
@@ -168,6 +168,20 @@ FUSE_3.7 {
                fuse_log;
 } FUSE_3.4;
 
+FUSE_3.12 {
+       global:
+               fuse_session_loop_mt;
+               fuse_session_loop_mt_312;
+               fuse_loop_mt;
+               fuse_loop_mt_32;
+               fuse_loop_mt_312;
+               fuse_loop_cfg_create;
+               fuse_loop_cfg_destroy;
+               fuse_loop_cfg_set_idle_threads;
+               fuse_loop_cfg_set_clone_fd;
+               fuse_loop_cfg_convert;
+} FUSE_3.4;
+
 # Local Variables:
 # indent-tabs-mode: t
 # End:
index 64ff7ad6d598b68182219b80921a16284cfd6db9..fc6a6ee1eb9cd5790163b1b3f6106727d554bb4a 100644 (file)
@@ -283,6 +283,7 @@ int fuse_main_real(int argc, char *argv[], const struct fuse_operations *op,
        struct fuse *fuse;
        struct fuse_cmdline_opts opts;
        int res;
+       struct fuse_loop_config *loop_config = NULL;
 
        if (fuse_parse_cmdline(&args, &opts) != 0)
                return 1;
@@ -338,13 +339,19 @@ int fuse_main_real(int argc, char *argv[], const struct fuse_operations *op,
        if (opts.singlethread)
                res = fuse_loop(fuse);
        else {
-               struct fuse_loop_config loop_config;
-               loop_config.clone_fd = opts.clone_fd;
-               loop_config.max_idle_threads = opts.max_idle_threads;
-               res = fuse_loop_mt_32(fuse, &loop_config);
+               loop_config = fuse_loop_cfg_create();
+               if (loop_config == NULL) {
+                       res = 7;
+                       goto out3;
+               }
+
+               fuse_loop_cfg_set_clone_fd(loop_config, opts.clone_fd);
+
+               fuse_loop_cfg_set_idle_threads(loop_config, opts.max_idle_threads);
+               res = fuse_loop_mt(fuse, loop_config);
        }
        if (res)
-               res = 7;
+               res = 8;
 
        fuse_remove_signal_handlers(se);
 out3:
@@ -352,6 +359,7 @@ out3:
 out2:
        fuse_destroy(fuse);
 out1:
+       fuse_loop_cfg_destroy(loop_config);
        free(opts.mountpoint);
        fuse_opt_free_args(&args);
        return res;
index 98461d8b0c22bdef4a1ec411de292a9ef74c786f..ef0e11ebde75c9356fc5144f6b3335eb6d569ab8 100644 (file)
@@ -37,7 +37,7 @@ libfuse = library('fuse3', libfuse_sources, version: meson.project_version(),
                   soversion: '3', include_directories: include_dirs,
                   dependencies: deps, install: true,
                   link_depends: 'fuse_versionscript',
-                  c_args: [ '-DFUSE_USE_VERSION=35',
+                  c_args: [ '-DFUSE_USE_VERSION=312',
                             '-DFUSERMOUNT_DIR="@0@"'.format(fusermount_path) ],
                   link_args: ['-Wl,--version-script,' + meson.current_source_dir()
                               + '/fuse_versionscript' ])
index 342dee1c1e37cfec5043ba050bcd6ff3e0a95ec8..3cef64fd11ea300e8416f539a12873485365d508 100644 (file)
@@ -1,4 +1,4 @@
-project('libfuse3', ['c'], version: '3.11.0',
+project('libfuse3', ['c'], version: '3.12.0',
         meson_version: '>= 0.42',
         default_options: [
             'buildtype=debugoptimized',
index 577668fd5800e8b0511097d49f2ea0bfdfc1a96f..78468706d8e17fe3997fe26afb6d0f9f9e6c057e 100644 (file)
@@ -11,7 +11,7 @@ executable('mount.fuse3', ['mount.fuse.c'],
            link_with: [ libfuse ],
            install: true,
            install_dir: get_option('sbindir'),
-           c_args: '-DFUSE_USE_VERSION=35')
+           c_args: '-DFUSE_USE_VERSION=312')
 
 
 udevrulesdir = get_option('udevrulesdir')