9b3b353ef3
Commit 9d682ea6bcc7 ("vboxsf: Fix the check for the old binary mount-arguments struct") was meant to fix a build error due to sign mismatch in 'char' and the use of character constants, but it just moved the error elsewhere, in that on some architectures characters and signed and on others they are unsigned, and that's just how the C standard works. The proper fix is a simple "don't do that then". The code was just being silly and odd, and it should never have cared about signed vs unsigned characters in the first place, since what it is testing is not four "characters", but four bytes. And the way to compare four bytes is by using "memcmp()". Which compilers will know to just turn into a single 32-bit compare with a constant, as long as you don't have crazy debug options enabled. Link: https://lore.kernel.org/lkml/20210927094123.576521-1-arnd@kernel.org/ Cc: Arnd Bergmann <arnd@kernel.org> Cc: Hans de Goede <hdegoede@redhat.com> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
486 lines
11 KiB
C
486 lines
11 KiB
C
// SPDX-License-Identifier: MIT
|
|
/*
|
|
* VirtualBox Guest Shared Folders support: Virtual File System.
|
|
*
|
|
* Module initialization/finalization
|
|
* File system registration/deregistration
|
|
* Superblock reading
|
|
* Few utility functions
|
|
*
|
|
* Copyright (C) 2006-2018 Oracle Corporation
|
|
*/
|
|
|
|
#include <linux/idr.h>
|
|
#include <linux/fs_parser.h>
|
|
#include <linux/magic.h>
|
|
#include <linux/module.h>
|
|
#include <linux/nls.h>
|
|
#include <linux/statfs.h>
|
|
#include <linux/vbox_utils.h>
|
|
#include "vfsmod.h"
|
|
|
|
#define VBOXSF_SUPER_MAGIC 0x786f4256 /* 'VBox' little endian */
|
|
|
|
static const unsigned char VBSF_MOUNT_SIGNATURE[4] = "\000\377\376\375";
|
|
|
|
static int follow_symlinks;
|
|
module_param(follow_symlinks, int, 0444);
|
|
MODULE_PARM_DESC(follow_symlinks,
|
|
"Let host resolve symlinks rather than showing them");
|
|
|
|
static DEFINE_IDA(vboxsf_bdi_ida);
|
|
static DEFINE_MUTEX(vboxsf_setup_mutex);
|
|
static bool vboxsf_setup_done;
|
|
static struct super_operations vboxsf_super_ops; /* forward declaration */
|
|
static struct kmem_cache *vboxsf_inode_cachep;
|
|
|
|
static char * const vboxsf_default_nls = CONFIG_NLS_DEFAULT;
|
|
|
|
enum { opt_nls, opt_uid, opt_gid, opt_ttl, opt_dmode, opt_fmode,
|
|
opt_dmask, opt_fmask };
|
|
|
|
static const struct fs_parameter_spec vboxsf_fs_parameters[] = {
|
|
fsparam_string ("nls", opt_nls),
|
|
fsparam_u32 ("uid", opt_uid),
|
|
fsparam_u32 ("gid", opt_gid),
|
|
fsparam_u32 ("ttl", opt_ttl),
|
|
fsparam_u32oct ("dmode", opt_dmode),
|
|
fsparam_u32oct ("fmode", opt_fmode),
|
|
fsparam_u32oct ("dmask", opt_dmask),
|
|
fsparam_u32oct ("fmask", opt_fmask),
|
|
{}
|
|
};
|
|
|
|
static int vboxsf_parse_param(struct fs_context *fc, struct fs_parameter *param)
|
|
{
|
|
struct vboxsf_fs_context *ctx = fc->fs_private;
|
|
struct fs_parse_result result;
|
|
kuid_t uid;
|
|
kgid_t gid;
|
|
int opt;
|
|
|
|
opt = fs_parse(fc, vboxsf_fs_parameters, param, &result);
|
|
if (opt < 0)
|
|
return opt;
|
|
|
|
switch (opt) {
|
|
case opt_nls:
|
|
if (ctx->nls_name || fc->purpose != FS_CONTEXT_FOR_MOUNT) {
|
|
vbg_err("vboxsf: Cannot reconfigure nls option\n");
|
|
return -EINVAL;
|
|
}
|
|
ctx->nls_name = param->string;
|
|
param->string = NULL;
|
|
break;
|
|
case opt_uid:
|
|
uid = make_kuid(current_user_ns(), result.uint_32);
|
|
if (!uid_valid(uid))
|
|
return -EINVAL;
|
|
ctx->o.uid = uid;
|
|
break;
|
|
case opt_gid:
|
|
gid = make_kgid(current_user_ns(), result.uint_32);
|
|
if (!gid_valid(gid))
|
|
return -EINVAL;
|
|
ctx->o.gid = gid;
|
|
break;
|
|
case opt_ttl:
|
|
ctx->o.ttl = msecs_to_jiffies(result.uint_32);
|
|
break;
|
|
case opt_dmode:
|
|
if (result.uint_32 & ~0777)
|
|
return -EINVAL;
|
|
ctx->o.dmode = result.uint_32;
|
|
ctx->o.dmode_set = true;
|
|
break;
|
|
case opt_fmode:
|
|
if (result.uint_32 & ~0777)
|
|
return -EINVAL;
|
|
ctx->o.fmode = result.uint_32;
|
|
ctx->o.fmode_set = true;
|
|
break;
|
|
case opt_dmask:
|
|
if (result.uint_32 & ~07777)
|
|
return -EINVAL;
|
|
ctx->o.dmask = result.uint_32;
|
|
break;
|
|
case opt_fmask:
|
|
if (result.uint_32 & ~07777)
|
|
return -EINVAL;
|
|
ctx->o.fmask = result.uint_32;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int vboxsf_fill_super(struct super_block *sb, struct fs_context *fc)
|
|
{
|
|
struct vboxsf_fs_context *ctx = fc->fs_private;
|
|
struct shfl_string *folder_name, root_path;
|
|
struct vboxsf_sbi *sbi;
|
|
struct dentry *droot;
|
|
struct inode *iroot;
|
|
char *nls_name;
|
|
size_t size;
|
|
int err;
|
|
|
|
if (!fc->source)
|
|
return -EINVAL;
|
|
|
|
sbi = kzalloc(sizeof(*sbi), GFP_KERNEL);
|
|
if (!sbi)
|
|
return -ENOMEM;
|
|
|
|
sbi->o = ctx->o;
|
|
idr_init(&sbi->ino_idr);
|
|
spin_lock_init(&sbi->ino_idr_lock);
|
|
sbi->next_generation = 1;
|
|
sbi->bdi_id = -1;
|
|
|
|
/* Load nls if not utf8 */
|
|
nls_name = ctx->nls_name ? ctx->nls_name : vboxsf_default_nls;
|
|
if (strcmp(nls_name, "utf8") != 0) {
|
|
if (nls_name == vboxsf_default_nls)
|
|
sbi->nls = load_nls_default();
|
|
else
|
|
sbi->nls = load_nls(nls_name);
|
|
|
|
if (!sbi->nls) {
|
|
vbg_err("vboxsf: Count not load '%s' nls\n", nls_name);
|
|
err = -EINVAL;
|
|
goto fail_free;
|
|
}
|
|
}
|
|
|
|
sbi->bdi_id = ida_simple_get(&vboxsf_bdi_ida, 0, 0, GFP_KERNEL);
|
|
if (sbi->bdi_id < 0) {
|
|
err = sbi->bdi_id;
|
|
goto fail_free;
|
|
}
|
|
|
|
err = super_setup_bdi_name(sb, "vboxsf-%d", sbi->bdi_id);
|
|
if (err)
|
|
goto fail_free;
|
|
sb->s_bdi->ra_pages = 0;
|
|
sb->s_bdi->io_pages = 0;
|
|
|
|
/* Turn source into a shfl_string and map the folder */
|
|
size = strlen(fc->source) + 1;
|
|
folder_name = kmalloc(SHFLSTRING_HEADER_SIZE + size, GFP_KERNEL);
|
|
if (!folder_name) {
|
|
err = -ENOMEM;
|
|
goto fail_free;
|
|
}
|
|
folder_name->size = size;
|
|
folder_name->length = size - 1;
|
|
strlcpy(folder_name->string.utf8, fc->source, size);
|
|
err = vboxsf_map_folder(folder_name, &sbi->root);
|
|
kfree(folder_name);
|
|
if (err) {
|
|
vbg_err("vboxsf: Host rejected mount of '%s' with error %d\n",
|
|
fc->source, err);
|
|
goto fail_free;
|
|
}
|
|
|
|
root_path.length = 1;
|
|
root_path.size = 2;
|
|
root_path.string.utf8[0] = '/';
|
|
root_path.string.utf8[1] = 0;
|
|
err = vboxsf_stat(sbi, &root_path, &sbi->root_info);
|
|
if (err)
|
|
goto fail_unmap;
|
|
|
|
sb->s_magic = VBOXSF_SUPER_MAGIC;
|
|
sb->s_blocksize = 1024;
|
|
sb->s_maxbytes = MAX_LFS_FILESIZE;
|
|
sb->s_op = &vboxsf_super_ops;
|
|
sb->s_d_op = &vboxsf_dentry_ops;
|
|
|
|
iroot = iget_locked(sb, 0);
|
|
if (!iroot) {
|
|
err = -ENOMEM;
|
|
goto fail_unmap;
|
|
}
|
|
vboxsf_init_inode(sbi, iroot, &sbi->root_info, false);
|
|
unlock_new_inode(iroot);
|
|
|
|
droot = d_make_root(iroot);
|
|
if (!droot) {
|
|
err = -ENOMEM;
|
|
goto fail_unmap;
|
|
}
|
|
|
|
sb->s_root = droot;
|
|
sb->s_fs_info = sbi;
|
|
return 0;
|
|
|
|
fail_unmap:
|
|
vboxsf_unmap_folder(sbi->root);
|
|
fail_free:
|
|
if (sbi->bdi_id >= 0)
|
|
ida_simple_remove(&vboxsf_bdi_ida, sbi->bdi_id);
|
|
if (sbi->nls)
|
|
unload_nls(sbi->nls);
|
|
idr_destroy(&sbi->ino_idr);
|
|
kfree(sbi);
|
|
return err;
|
|
}
|
|
|
|
static void vboxsf_inode_init_once(void *data)
|
|
{
|
|
struct vboxsf_inode *sf_i = data;
|
|
|
|
mutex_init(&sf_i->handle_list_mutex);
|
|
inode_init_once(&sf_i->vfs_inode);
|
|
}
|
|
|
|
static struct inode *vboxsf_alloc_inode(struct super_block *sb)
|
|
{
|
|
struct vboxsf_inode *sf_i;
|
|
|
|
sf_i = kmem_cache_alloc(vboxsf_inode_cachep, GFP_NOFS);
|
|
if (!sf_i)
|
|
return NULL;
|
|
|
|
sf_i->force_restat = 0;
|
|
INIT_LIST_HEAD(&sf_i->handle_list);
|
|
|
|
return &sf_i->vfs_inode;
|
|
}
|
|
|
|
static void vboxsf_free_inode(struct inode *inode)
|
|
{
|
|
struct vboxsf_sbi *sbi = VBOXSF_SBI(inode->i_sb);
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&sbi->ino_idr_lock, flags);
|
|
idr_remove(&sbi->ino_idr, inode->i_ino);
|
|
spin_unlock_irqrestore(&sbi->ino_idr_lock, flags);
|
|
kmem_cache_free(vboxsf_inode_cachep, VBOXSF_I(inode));
|
|
}
|
|
|
|
static void vboxsf_put_super(struct super_block *sb)
|
|
{
|
|
struct vboxsf_sbi *sbi = VBOXSF_SBI(sb);
|
|
|
|
vboxsf_unmap_folder(sbi->root);
|
|
if (sbi->bdi_id >= 0)
|
|
ida_simple_remove(&vboxsf_bdi_ida, sbi->bdi_id);
|
|
if (sbi->nls)
|
|
unload_nls(sbi->nls);
|
|
|
|
/*
|
|
* vboxsf_free_inode uses the idr, make sure all delayed rcu free
|
|
* inodes are flushed.
|
|
*/
|
|
rcu_barrier();
|
|
idr_destroy(&sbi->ino_idr);
|
|
kfree(sbi);
|
|
}
|
|
|
|
static int vboxsf_statfs(struct dentry *dentry, struct kstatfs *stat)
|
|
{
|
|
struct super_block *sb = dentry->d_sb;
|
|
struct shfl_volinfo shfl_volinfo;
|
|
struct vboxsf_sbi *sbi;
|
|
u32 buf_len;
|
|
int err;
|
|
|
|
sbi = VBOXSF_SBI(sb);
|
|
buf_len = sizeof(shfl_volinfo);
|
|
err = vboxsf_fsinfo(sbi->root, 0, SHFL_INFO_GET | SHFL_INFO_VOLUME,
|
|
&buf_len, &shfl_volinfo);
|
|
if (err)
|
|
return err;
|
|
|
|
stat->f_type = VBOXSF_SUPER_MAGIC;
|
|
stat->f_bsize = shfl_volinfo.bytes_per_allocation_unit;
|
|
|
|
do_div(shfl_volinfo.total_allocation_bytes,
|
|
shfl_volinfo.bytes_per_allocation_unit);
|
|
stat->f_blocks = shfl_volinfo.total_allocation_bytes;
|
|
|
|
do_div(shfl_volinfo.available_allocation_bytes,
|
|
shfl_volinfo.bytes_per_allocation_unit);
|
|
stat->f_bfree = shfl_volinfo.available_allocation_bytes;
|
|
stat->f_bavail = shfl_volinfo.available_allocation_bytes;
|
|
|
|
stat->f_files = 1000;
|
|
/*
|
|
* Don't return 0 here since the guest may then think that it is not
|
|
* possible to create any more files.
|
|
*/
|
|
stat->f_ffree = 1000000;
|
|
stat->f_fsid.val[0] = 0;
|
|
stat->f_fsid.val[1] = 0;
|
|
stat->f_namelen = 255;
|
|
return 0;
|
|
}
|
|
|
|
static struct super_operations vboxsf_super_ops = {
|
|
.alloc_inode = vboxsf_alloc_inode,
|
|
.free_inode = vboxsf_free_inode,
|
|
.put_super = vboxsf_put_super,
|
|
.statfs = vboxsf_statfs,
|
|
};
|
|
|
|
static int vboxsf_setup(void)
|
|
{
|
|
int err;
|
|
|
|
mutex_lock(&vboxsf_setup_mutex);
|
|
|
|
if (vboxsf_setup_done)
|
|
goto success;
|
|
|
|
vboxsf_inode_cachep =
|
|
kmem_cache_create("vboxsf_inode_cache",
|
|
sizeof(struct vboxsf_inode), 0,
|
|
(SLAB_RECLAIM_ACCOUNT | SLAB_MEM_SPREAD |
|
|
SLAB_ACCOUNT),
|
|
vboxsf_inode_init_once);
|
|
if (!vboxsf_inode_cachep) {
|
|
err = -ENOMEM;
|
|
goto fail_nomem;
|
|
}
|
|
|
|
err = vboxsf_connect();
|
|
if (err) {
|
|
vbg_err("vboxsf: err %d connecting to guest PCI-device\n", err);
|
|
vbg_err("vboxsf: make sure you are inside a VirtualBox VM\n");
|
|
vbg_err("vboxsf: and check dmesg for vboxguest errors\n");
|
|
goto fail_free_cache;
|
|
}
|
|
|
|
err = vboxsf_set_utf8();
|
|
if (err) {
|
|
vbg_err("vboxsf_setutf8 error %d\n", err);
|
|
goto fail_disconnect;
|
|
}
|
|
|
|
if (!follow_symlinks) {
|
|
err = vboxsf_set_symlinks();
|
|
if (err)
|
|
vbg_warn("vboxsf: Unable to show symlinks: %d\n", err);
|
|
}
|
|
|
|
vboxsf_setup_done = true;
|
|
success:
|
|
mutex_unlock(&vboxsf_setup_mutex);
|
|
return 0;
|
|
|
|
fail_disconnect:
|
|
vboxsf_disconnect();
|
|
fail_free_cache:
|
|
kmem_cache_destroy(vboxsf_inode_cachep);
|
|
fail_nomem:
|
|
mutex_unlock(&vboxsf_setup_mutex);
|
|
return err;
|
|
}
|
|
|
|
static int vboxsf_parse_monolithic(struct fs_context *fc, void *data)
|
|
{
|
|
if (data && !memcmp(data, VBSF_MOUNT_SIGNATURE, 4)) {
|
|
vbg_err("vboxsf: Old binary mount data not supported, remove obsolete mount.vboxsf and/or update your VBoxService.\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
return generic_parse_monolithic(fc, data);
|
|
}
|
|
|
|
static int vboxsf_get_tree(struct fs_context *fc)
|
|
{
|
|
int err;
|
|
|
|
err = vboxsf_setup();
|
|
if (err)
|
|
return err;
|
|
|
|
return get_tree_nodev(fc, vboxsf_fill_super);
|
|
}
|
|
|
|
static int vboxsf_reconfigure(struct fs_context *fc)
|
|
{
|
|
struct vboxsf_sbi *sbi = VBOXSF_SBI(fc->root->d_sb);
|
|
struct vboxsf_fs_context *ctx = fc->fs_private;
|
|
struct inode *iroot = fc->root->d_sb->s_root->d_inode;
|
|
|
|
/* Apply changed options to the root inode */
|
|
sbi->o = ctx->o;
|
|
vboxsf_init_inode(sbi, iroot, &sbi->root_info, true);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void vboxsf_free_fc(struct fs_context *fc)
|
|
{
|
|
struct vboxsf_fs_context *ctx = fc->fs_private;
|
|
|
|
kfree(ctx->nls_name);
|
|
kfree(ctx);
|
|
}
|
|
|
|
static const struct fs_context_operations vboxsf_context_ops = {
|
|
.free = vboxsf_free_fc,
|
|
.parse_param = vboxsf_parse_param,
|
|
.parse_monolithic = vboxsf_parse_monolithic,
|
|
.get_tree = vboxsf_get_tree,
|
|
.reconfigure = vboxsf_reconfigure,
|
|
};
|
|
|
|
static int vboxsf_init_fs_context(struct fs_context *fc)
|
|
{
|
|
struct vboxsf_fs_context *ctx;
|
|
|
|
ctx = kzalloc(sizeof(*ctx), GFP_KERNEL);
|
|
if (!ctx)
|
|
return -ENOMEM;
|
|
|
|
current_uid_gid(&ctx->o.uid, &ctx->o.gid);
|
|
|
|
fc->fs_private = ctx;
|
|
fc->ops = &vboxsf_context_ops;
|
|
return 0;
|
|
}
|
|
|
|
static struct file_system_type vboxsf_fs_type = {
|
|
.owner = THIS_MODULE,
|
|
.name = "vboxsf",
|
|
.init_fs_context = vboxsf_init_fs_context,
|
|
.kill_sb = kill_anon_super
|
|
};
|
|
|
|
/* Module initialization/finalization handlers */
|
|
static int __init vboxsf_init(void)
|
|
{
|
|
return register_filesystem(&vboxsf_fs_type);
|
|
}
|
|
|
|
static void __exit vboxsf_fini(void)
|
|
{
|
|
unregister_filesystem(&vboxsf_fs_type);
|
|
|
|
mutex_lock(&vboxsf_setup_mutex);
|
|
if (vboxsf_setup_done) {
|
|
vboxsf_disconnect();
|
|
/*
|
|
* Make sure all delayed rcu free inodes are flushed
|
|
* before we destroy the cache.
|
|
*/
|
|
rcu_barrier();
|
|
kmem_cache_destroy(vboxsf_inode_cachep);
|
|
}
|
|
mutex_unlock(&vboxsf_setup_mutex);
|
|
}
|
|
|
|
module_init(vboxsf_init);
|
|
module_exit(vboxsf_fini);
|
|
|
|
MODULE_DESCRIPTION("Oracle VM VirtualBox Module for Host File System Access");
|
|
MODULE_AUTHOR("Oracle Corporation");
|
|
MODULE_LICENSE("GPL v2");
|
|
MODULE_ALIAS_FS("vboxsf");
|