提交 a7f0d7aa 编写于 作者: C Connor Kuehl 提交者: Miklos Szeredi

virtiofs: split requests that exceed virtqueue size

If an incoming FUSE request can't fit on the virtqueue, the request is
placed onto a workqueue so a worker can try to resubmit it later where
there will (hopefully) be space for it next time.

This is fine for requests that aren't larger than a virtqueue's maximum
capacity.  However, if a request's size exceeds the maximum capacity of the
virtqueue (even if the virtqueue is empty), it will be doomed to a life of
being placed on the workqueue, removed, discovered it won't fit, and placed
on the workqueue yet again.

Furthermore, from section 2.6.5.3.1 (Driver Requirements: Indirect
Descriptors) of the virtio spec:

  "A driver MUST NOT create a descriptor chain longer than the Queue
  Size of the device."

To fix this, limit the number of pages FUSE will use for an overall
request.  This way, each request can realistically fit on the virtqueue
when it is decomposed into a scattergather list and avoid violating section
2.6.5.3.1 of the virtio spec.
Signed-off-by: NConnor Kuehl <ckuehl@redhat.com>
Reviewed-by: NVivek Goyal <vgoyal@redhat.com>
Signed-off-by: NMiklos Szeredi <mszeredi@redhat.com>
上级 c79c5e01
...@@ -555,6 +555,9 @@ struct fuse_conn { ...@@ -555,6 +555,9 @@ struct fuse_conn {
/** Maximum number of pages that can be used in a single request */ /** Maximum number of pages that can be used in a single request */
unsigned int max_pages; unsigned int max_pages;
/** Constrain ->max_pages to this value during feature negotiation */
unsigned int max_pages_limit;
/** Input queue */ /** Input queue */
struct fuse_iqueue iq; struct fuse_iqueue iq;
......
...@@ -712,6 +712,7 @@ void fuse_conn_init(struct fuse_conn *fc, struct fuse_mount *fm, ...@@ -712,6 +712,7 @@ void fuse_conn_init(struct fuse_conn *fc, struct fuse_mount *fm,
fc->pid_ns = get_pid_ns(task_active_pid_ns(current)); fc->pid_ns = get_pid_ns(task_active_pid_ns(current));
fc->user_ns = get_user_ns(user_ns); fc->user_ns = get_user_ns(user_ns);
fc->max_pages = FUSE_DEFAULT_MAX_PAGES_PER_REQ; fc->max_pages = FUSE_DEFAULT_MAX_PAGES_PER_REQ;
fc->max_pages_limit = FUSE_MAX_MAX_PAGES;
INIT_LIST_HEAD(&fc->mounts); INIT_LIST_HEAD(&fc->mounts);
list_add(&fm->fc_entry, &fc->mounts); list_add(&fm->fc_entry, &fc->mounts);
...@@ -1040,7 +1041,7 @@ static void process_init_reply(struct fuse_mount *fm, struct fuse_args *args, ...@@ -1040,7 +1041,7 @@ static void process_init_reply(struct fuse_mount *fm, struct fuse_args *args,
fc->abort_err = 1; fc->abort_err = 1;
if (arg->flags & FUSE_MAX_PAGES) { if (arg->flags & FUSE_MAX_PAGES) {
fc->max_pages = fc->max_pages =
min_t(unsigned int, FUSE_MAX_MAX_PAGES, min_t(unsigned int, fc->max_pages_limit,
max_t(unsigned int, arg->max_pages, 1)); max_t(unsigned int, arg->max_pages, 1));
} }
if (IS_ENABLED(CONFIG_FUSE_DAX) && if (IS_ENABLED(CONFIG_FUSE_DAX) &&
......
...@@ -18,6 +18,12 @@ ...@@ -18,6 +18,12 @@
#include <linux/uio.h> #include <linux/uio.h>
#include "fuse_i.h" #include "fuse_i.h"
/* Used to help calculate the FUSE connection's max_pages limit for a request's
* size. Parts of the struct fuse_req are sliced into scattergather lists in
* addition to the pages used, so this can help account for that overhead.
*/
#define FUSE_HEADER_OVERHEAD 4
/* List of virtio-fs device instances and a lock for the list. Also provides /* List of virtio-fs device instances and a lock for the list. Also provides
* mutual exclusion in device removal and mounting path * mutual exclusion in device removal and mounting path
*/ */
...@@ -1414,9 +1420,10 @@ static int virtio_fs_get_tree(struct fs_context *fsc) ...@@ -1414,9 +1420,10 @@ static int virtio_fs_get_tree(struct fs_context *fsc)
{ {
struct virtio_fs *fs; struct virtio_fs *fs;
struct super_block *sb; struct super_block *sb;
struct fuse_conn *fc; struct fuse_conn *fc = NULL;
struct fuse_mount *fm; struct fuse_mount *fm;
int err; unsigned int virtqueue_size;
int err = -EIO;
/* This gets a reference on virtio_fs object. This ptr gets installed /* This gets a reference on virtio_fs object. This ptr gets installed
* in fc->iq->priv. Once fuse_conn is going away, it calls ->put() * in fc->iq->priv. Once fuse_conn is going away, it calls ->put()
...@@ -1428,6 +1435,10 @@ static int virtio_fs_get_tree(struct fs_context *fsc) ...@@ -1428,6 +1435,10 @@ static int virtio_fs_get_tree(struct fs_context *fsc)
return -EINVAL; return -EINVAL;
} }
virtqueue_size = virtqueue_get_vring_size(fs->vqs[VQ_REQUEST].vq);
if (WARN_ON(virtqueue_size <= FUSE_HEADER_OVERHEAD))
goto out_err;
err = -ENOMEM; err = -ENOMEM;
fc = kzalloc(sizeof(struct fuse_conn), GFP_KERNEL); fc = kzalloc(sizeof(struct fuse_conn), GFP_KERNEL);
if (!fc) if (!fc)
...@@ -1443,6 +1454,10 @@ static int virtio_fs_get_tree(struct fs_context *fsc) ...@@ -1443,6 +1454,10 @@ static int virtio_fs_get_tree(struct fs_context *fsc)
fc->delete_stale = true; fc->delete_stale = true;
fc->auto_submounts = true; fc->auto_submounts = true;
/* Tell FUSE to split requests that exceed the virtqueue's size */
fc->max_pages_limit = min_t(unsigned int, fc->max_pages_limit,
virtqueue_size - FUSE_HEADER_OVERHEAD);
fsc->s_fs_info = fm; fsc->s_fs_info = fm;
sb = sget_fc(fsc, virtio_fs_test_super, set_anon_super_fc); sb = sget_fc(fsc, virtio_fs_test_super, set_anon_super_fc);
if (fsc->s_fs_info) { if (fsc->s_fs_info) {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册