Skip to content

Commit

Permalink
virtiofs: Do not use fuse_fill_super_common() for fuse device install…
Browse files Browse the repository at this point in the history
…ation

fuse_fill_super_common() allocates and installs one fuse_device. Hence
virtiofs allocates and install all fuse devices by itself except one.

This makes logic little twisted. There does not seem to be any real need
that why virtiofs can't allocate and install all fuse devices itself. 

So opt out of fuse device allocation and installation while calling
fuse_fill_super_common().

Regular fuse still wants fuse_fill_super_common() to install fuse_device.
It needs to prevent against races where two mounters are trying to mount
fuse using same fd. In that case one will succeed while other will get
-EINVAL. 

virtiofs does not have this issue because sget_fc() resolves the race
w.r.t multiple mounters and only one instance of virtio_fs_fill_super()
should be in progress for same filesystem.

Signed-off-by: Vivek Goyal <vgoyal@redhat.com>
  • Loading branch information
rhvgoyal committed May 4, 2020
1 parent 6a8b55e commit 29eb772
Show file tree
Hide file tree
Showing 2 changed files with 15 additions and 13 deletions.
19 changes: 12 additions & 7 deletions fs/fuse/inode.c
Expand Up @@ -1113,7 +1113,7 @@ EXPORT_SYMBOL_GPL(fuse_dev_free);

int fuse_fill_super_common(struct super_block *sb, struct fuse_fs_context *ctx)
{
struct fuse_dev *fud;
struct fuse_dev *fud = NULL;
struct fuse_conn *fc = get_fuse_conn_super(sb);
struct inode *root;
struct dentry *root_dentry;
Expand Down Expand Up @@ -1155,9 +1155,12 @@ int fuse_fill_super_common(struct super_block *sb, struct fuse_fs_context *ctx)
if (sb->s_user_ns != &init_user_ns)
sb->s_xattr = fuse_no_acl_xattr_handlers;

fud = fuse_dev_alloc_install(fc);
if (!fud)
goto err;
if (ctx->fudptr) {
err = -ENOMEM;
fud = fuse_dev_alloc_install(fc);
if (!fud)
goto err;
}

fc->dev = sb->s_dev;
fc->sb = sb;
Expand Down Expand Up @@ -1191,7 +1194,7 @@ int fuse_fill_super_common(struct super_block *sb, struct fuse_fs_context *ctx)

mutex_lock(&fuse_mutex);
err = -EINVAL;
if (*ctx->fudptr)
if (ctx->fudptr && *ctx->fudptr)
goto err_unlock;

err = fuse_ctl_add_conn(fc);
Expand All @@ -1200,15 +1203,17 @@ int fuse_fill_super_common(struct super_block *sb, struct fuse_fs_context *ctx)

list_add_tail(&fc->entry, &fuse_conn_list);
sb->s_root = root_dentry;
*ctx->fudptr = fud;
if (ctx->fudptr)
*ctx->fudptr = fud;
mutex_unlock(&fuse_mutex);
return 0;

err_unlock:
mutex_unlock(&fuse_mutex);
dput(root_dentry);
err_dev_free:
fuse_dev_free(fud);
if (fud)
fuse_dev_free(fud);
err:
return err;
}
Expand Down
9 changes: 3 additions & 6 deletions fs/fuse/virtio_fs.c
Expand Up @@ -1067,26 +1067,23 @@ static int virtio_fs_fill_super(struct super_block *sb)

err = -ENOMEM;
/* Allocate fuse_dev for hiprio and notification queues */
for (i = 0; i < VQ_REQUEST; i++) {
for (i = 0; i < fs->nvqs; i++) {
struct virtio_fs_vq *fsvq = &fs->vqs[i];

fsvq->fud = fuse_dev_alloc();
if (!fsvq->fud)
goto err_free_fuse_devs;
}

ctx.fudptr = (void **)&fs->vqs[VQ_REQUEST].fud;
/* virtiofs allocates and installs its own fuse devices */
ctx.fudptr = NULL;
err = fuse_fill_super_common(sb, &ctx);
if (err < 0)
goto err_free_fuse_devs;

fc = fs->vqs[VQ_REQUEST].fud->fc;

for (i = 0; i < fs->nvqs; i++) {
struct virtio_fs_vq *fsvq = &fs->vqs[i];

if (i == VQ_REQUEST)
continue; /* already initialized */
fuse_dev_install(fsvq->fud, fc);
}

Expand Down

0 comments on commit 29eb772

Please sign in to comment.