Unreleased Changes
 ==================
 
+* Added a `max_read` field to `struct fuse_conn_info`. For the time
+  being, the maximum size of read requests has to be specified both
+  there *and* passed to fuse_session_new() using the ``-o
+  max_read=<n>`` mount option. At some point in the future, specifying
+  the mount option will no longer be necessary.
+
 * Documentation: clarified that the fuse_argv structure that is passed
   to `fuse_new()` and `fuse_lowlevel_new()` must always contain at
   least one element.
 
 \fBmax_read=N\fP
 With this option the maximum size of read operations can be set. The
 default is infinite, but typically the kernel enforces its own limit
-in addition to this one.
+in addition to this one. A value of zero corresponds to no limit.
 
 This option should not be specified by the filesystem owner. The
 correct (or optimum) value depends on the filesystem implementation
 and should thus be set by the filesystem internally.
+
+This mount option is deprecated in favor of direct negotiation over
+the device fd (as done for e.g. the maximum size of write
+operations). For the time being, libfuse-using filesystems that want
+to limit the read size must therefore use this mount option \fIand\fP
+set the same value again in the init() handler.
 .TP
 \fBfd=N\fP
 The file descriptor to use for communication between the userspace
 
         */
        unsigned max_write;
 
+       /**
+        * Maximum size of read requests. A value of zero indicates no
+        * limit. However, even if the filesystem does not specify a
+        * limit, the maximum size of read requests will still be
+        * limited by the kernel.
+        *
+        * NOTE: For the time being, the maximum size of read requests
+        * must be set both here *and* passed to fuse_session_new()
+        * using the ``-o max_read=<n>`` mount option. At some point
+        * in the future, specifying the mount option will no longer
+        * be necessary.
+        */
+       unsigned max_read;
+
        /**
         * Maximum readahead
         */
 
 struct mount_opts *parse_mount_opts(struct fuse_args *args);
 void destroy_mount_opts(struct mount_opts *mo);
 void fuse_mount_version(void);
+unsigned get_max_read(struct mount_opts *o);
 void fuse_kern_unmount(const char *mountpoint, int fd);
 int fuse_kern_mount(const char *mountpoint, struct mount_opts *mo);
 
 
        if (se->op.init)
                se->op.init(se->userdata, &se->conn);
 
+       unsigned max_read_mo = get_max_read(se->mo);
+       if (se->conn.max_read != max_read_mo) {
+               fprintf(stderr, "fuse: error: init() and fuse_session_new() "
+                       "requested different maximum read size (%u vs %u)\n",
+                       se->conn.max_read, max_read_mo);
+               abort();
+       }
+
        /* Always enable big writes, this is superseded
           by the max_write option */
        outarg.flags |= FUSE_BIG_WRITES;
 
        char *mtab_opts;
        char *fusermount_opts;
        char *kernel_opts;
+       unsigned max_read;
 };
 
 #define FUSE_MOUNT_OPT(t, p) { t, offsetof(struct mount_opts, p), 1 }
        FUSE_MOUNT_OPT("blkdev",                blkdev),
        FUSE_MOUNT_OPT("auto_unmount",          auto_unmount),
        FUSE_MOUNT_OPT("fsname=%s",             fsname),
+       FUSE_MOUNT_OPT("max_read=%u",           max_read),
        FUSE_MOUNT_OPT("subtype=%s",            subtype),
        FUSE_OPT_KEY("allow_other",             KEY_KERN_OPT),
        FUSE_OPT_KEY("allow_root",              KEY_ALLOW_ROOT),
        {NULL,      0,              0}
 };
 
+unsigned get_max_read(struct mount_opts *o)
+{
+       return o->max_read;
+}
+
 static void set_mount_flag(const char *s, int *flags)
 {
        int i;
 
        int allow_other;
        int allow_root;
        char *kernel_opts;
+       unsigned max_read;
 };
 
 #define FUSE_DUAL_OPT_KEY(templ, key)                          \
 static const struct fuse_opt fuse_mount_opts[] = {
        { "allow_other", offsetof(struct mount_opts, allow_other), 1 },
        { "allow_root", offsetof(struct mount_opts, allow_root), 1 },
+       { "max_read=%u", offsetof(struct mount_opts, max_read), 1 },
        FUSE_OPT_KEY("allow_root",              KEY_ALLOW_ROOT),
        FUSE_OPT_KEY("-r",                      KEY_RO),
        /* standard FreeBSD mount options */
        system(FUSERMOUNT_PROG " --version");
 }
 
+unsigned get_max_read(struct mount_opts *o)
+{
+       return o->max_read;
+}
+
 static int fuse_mount_opt_proc(void *data, const char *arg, int key,
                               struct fuse_args *outargs)
 {