Commit d6392f87 authored by Miklos Szeredi's avatar Miklos Szeredi Committed by Linus Torvalds

[PATCH] fuse: add support for block device based filesystems

I never intended this, but people started using fuse to implement block device
based "real" filesystems (ntfs-3g, zfs).

The following four patches add better support for these kinds of filesystems.
Unlike "normal" fuse filesystems, using this feature should require superuser
privileges (enforced by the fusermount utility).

Thanks to Szabolcs Szakacsits for the input and testing.

This patch adds a 'fuseblk' filesystem type, which is only different from the
'fuse' filesystem type in how the 'dev_name' mount argument is interpreted.
Signed-off-by: default avatarMiklos Szeredi <miklos@szeredi.hu>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent bdcf2508
...@@ -51,6 +51,22 @@ homepage: ...@@ -51,6 +51,22 @@ homepage:
http://fuse.sourceforge.net/ http://fuse.sourceforge.net/
Filesystem type
~~~~~~~~~~~~~~~
The filesystem type given to mount(2) can be one of the following:
'fuse'
This is the usual way to mount a FUSE filesystem. The first
argument of the mount system call may contain an arbitrary string,
which is not interpreted by the kernel.
'fuseblk'
The filesystem is block device based. The first argument of the
mount system call is interpreted as the name of the device.
Mount options Mount options
~~~~~~~~~~~~~ ~~~~~~~~~~~~~
......
...@@ -591,6 +591,14 @@ static int fuse_get_sb(struct file_system_type *fs_type, ...@@ -591,6 +591,14 @@ static int fuse_get_sb(struct file_system_type *fs_type,
return get_sb_nodev(fs_type, flags, raw_data, fuse_fill_super, mnt); return get_sb_nodev(fs_type, flags, raw_data, fuse_fill_super, mnt);
} }
static int fuse_get_sb_blk(struct file_system_type *fs_type,
int flags, const char *dev_name,
void *raw_data, struct vfsmount *mnt)
{
return get_sb_bdev(fs_type, flags, dev_name, raw_data, fuse_fill_super,
mnt);
}
static struct file_system_type fuse_fs_type = { static struct file_system_type fuse_fs_type = {
.owner = THIS_MODULE, .owner = THIS_MODULE,
.name = "fuse", .name = "fuse",
...@@ -598,6 +606,14 @@ static struct file_system_type fuse_fs_type = { ...@@ -598,6 +606,14 @@ static struct file_system_type fuse_fs_type = {
.kill_sb = kill_anon_super, .kill_sb = kill_anon_super,
}; };
static struct file_system_type fuseblk_fs_type = {
.owner = THIS_MODULE,
.name = "fuseblk",
.get_sb = fuse_get_sb_blk,
.kill_sb = kill_block_super,
.fs_flags = FS_REQUIRES_DEV,
};
static decl_subsys(fuse, NULL, NULL); static decl_subsys(fuse, NULL, NULL);
static decl_subsys(connections, NULL, NULL); static decl_subsys(connections, NULL, NULL);
...@@ -617,24 +633,34 @@ static int __init fuse_fs_init(void) ...@@ -617,24 +633,34 @@ static int __init fuse_fs_init(void)
err = register_filesystem(&fuse_fs_type); err = register_filesystem(&fuse_fs_type);
if (err) if (err)
printk("fuse: failed to register filesystem\n"); goto out;
else {
fuse_inode_cachep = kmem_cache_create("fuse_inode", err = register_filesystem(&fuseblk_fs_type);
sizeof(struct fuse_inode), if (err)
0, SLAB_HWCACHE_ALIGN, goto out_unreg;
fuse_inode_init_once, NULL);
if (!fuse_inode_cachep) { fuse_inode_cachep = kmem_cache_create("fuse_inode",
unregister_filesystem(&fuse_fs_type); sizeof(struct fuse_inode),
err = -ENOMEM; 0, SLAB_HWCACHE_ALIGN,
} fuse_inode_init_once, NULL);
} err = -ENOMEM;
if (!fuse_inode_cachep)
goto out_unreg2;
return 0;
out_unreg2:
unregister_filesystem(&fuseblk_fs_type);
out_unreg:
unregister_filesystem(&fuse_fs_type);
out:
return err; return err;
} }
static void fuse_fs_cleanup(void) static void fuse_fs_cleanup(void)
{ {
unregister_filesystem(&fuse_fs_type); unregister_filesystem(&fuse_fs_type);
unregister_filesystem(&fuseblk_fs_type);
kmem_cache_destroy(fuse_inode_cachep); kmem_cache_destroy(fuse_inode_cachep);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment